| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | using System; | 
					
						
							|  |  |  | using System.Net; | 
					
						
							|  |  |  | using System.Net.Sockets; | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | using System.Collections.Generic; | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace MonoTests.Helpers { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	public static class NetworkHelpers | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 		static Random rndPort = new Random (); | 
					
						
							|  |  |  | 		static HashSet<int> portsTable = new HashSet<int> (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 		public static int FindFreePort () | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 			return LocalEphemeralEndPoint ().Port; | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 		public static IPEndPoint LocalEphemeralEndPoint () | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 			int counter = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			while (counter < 1000) { | 
					
						
							|  |  |  | 				var testingPort = rndPort.Next (10000, 60000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				var ep = new IPEndPoint (IPAddress.Loopback, testingPort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				lock (portsTable) { | 
					
						
							|  |  |  | 					if (portsTable.Contains (testingPort)) | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					++counter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					try { | 
					
						
							|  |  |  | 						using (var socket = new Socket (ep.AddressFamily, SocketType.Stream, ProtocolType.Tcp)) { | 
					
						
							|  |  |  | 							socket.Bind (ep); | 
					
						
							|  |  |  | 							socket.Close (); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						portsTable.Add (testingPort); | 
					
						
							|  |  |  | 						return ep; | 
					
						
							|  |  |  | 					} catch (SocketException) { } | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			throw new ApplicationException ($"Could not find available local port after {counter} retries"); | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-04-12 14:10:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Bind to the specified address using a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the assigned port. | 
					
						
							|  |  |  | 		public static void Bind (this Socket socket, IPAddress address, out int port) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			socket.Bind (new IPEndPoint (address, 0)); | 
					
						
							|  |  |  | 			port = ((IPEndPoint) socket.LocalEndPoint).Port; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Bind to the specified address using a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the resulting end local end point. | 
					
						
							|  |  |  | 		public static void Bind (this Socket socket, IPAddress address, out IPEndPoint ep) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			socket.Bind (new IPEndPoint (address, 0)); | 
					
						
							|  |  |  | 			ep = (IPEndPoint) socket.LocalEndPoint; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts a TcpListener using a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the assigned port. | 
					
						
							|  |  |  | 		public static TcpListener CreateAndStartTcpListener (out int port) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var rv = new TcpListener (0); | 
					
						
							|  |  |  | 			rv.Start (); | 
					
						
							|  |  |  | 			port = ((IPEndPoint) rv.LocalEndpoint).Port; | 
					
						
							|  |  |  | 			return rv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts a TcpListener using a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the resulting local end point. | 
					
						
							|  |  |  | 		public static TcpListener CreateAndStartTcpListener (out IPEndPoint ep) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var rv = new TcpListener (0); | 
					
						
							|  |  |  | 			rv.Start (); | 
					
						
							|  |  |  | 			ep = (IPEndPoint) rv.LocalEndpoint; | 
					
						
							|  |  |  | 			return rv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts a TcpListener using the specified address and a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the assigned port. | 
					
						
							|  |  |  | 		public static TcpListener CreateAndStartTcpListener (IPAddress address, out int port) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var rv = new TcpListener (address, 0); | 
					
						
							|  |  |  | 			rv.Start (); | 
					
						
							|  |  |  | 			port = ((IPEndPoint) rv.LocalEndpoint).Port; | 
					
						
							|  |  |  | 			return rv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts a TcpListener using the specified address and a system-assigned port. | 
					
						
							|  |  |  | 		// Returns the resulting local end point. | 
					
						
							|  |  |  | 		public static TcpListener CreateAndStartTcpListener (IPAddress address, out IPEndPoint ep) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var rv = new TcpListener (address, 0); | 
					
						
							|  |  |  | 			rv.Start (); | 
					
						
							|  |  |  | 			ep = (IPEndPoint) rv.LocalEndpoint; | 
					
						
							|  |  |  | 			return rv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts an HttpListener using the specified host, port, | 
					
						
							|  |  |  | 		// path and authSchemes. | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// If specified, the initializer will be called immediately after the | 
					
						
							|  |  |  | 		// HttpListener is created (typical usage would be to set/change | 
					
						
							|  |  |  | 		// properties before starting the listener) | 
					
						
							|  |  |  | 		public static HttpListener CreateAndStartHttpListener (string host, int port, string path, AuthenticationSchemes? authSchemes = null, Action<HttpListener> initializer = null) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var prefix = host + port + path; | 
					
						
							|  |  |  | 			HttpListener listener = new HttpListener (); | 
					
						
							|  |  |  | 			if (initializer != null) | 
					
						
							|  |  |  | 				initializer (listener); | 
					
						
							|  |  |  | 			if (authSchemes.HasValue) | 
					
						
							|  |  |  | 				listener.AuthenticationSchemes = authSchemes.Value; | 
					
						
							|  |  |  | 			listener.Prefixes.Add (prefix); | 
					
						
							|  |  |  | 			listener.Start (); | 
					
						
							|  |  |  | 			return listener; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts an HttpListener using the specified host, path | 
					
						
							|  |  |  | 		// and authSchemes. The method will try to find an unused port, and | 
					
						
							|  |  |  | 		// use that (multiple attempts with random port numbers will be made). | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// If specified, the initializer will be called immediately after the | 
					
						
							|  |  |  | 		// HttpListener is created (typical usage would be to set/change | 
					
						
							|  |  |  | 		// properties before starting the listener). Be aware that the | 
					
						
							|  |  |  | 		// initializer can be called multiple times (in case multiple creation | 
					
						
							|  |  |  | 		// attempts have to be made). | 
					
						
							|  |  |  | 		public static HttpListener CreateAndStartHttpListener (string host, out int port, string path, AuthenticationSchemes? authSchemes = null, Action<HttpListener> initializer = null) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			// There's no way to create an HttpListener with a system-assigned port. | 
					
						
							|  |  |  | 			// So we use NetworkHelpers.FindFreePort, and re-try if we fail because someone else has already used the port. | 
					
						
							|  |  |  | 			for (int i = 0; i < 10; i++) { | 
					
						
							|  |  |  | 				try { | 
					
						
							|  |  |  | 					var tentativePort = NetworkHelpers.FindFreePort (); | 
					
						
							|  |  |  | 					var listener = CreateAndStartHttpListener (host, tentativePort, path, authSchemes, initializer); | 
					
						
							|  |  |  | 					port = tentativePort; | 
					
						
							|  |  |  | 					return listener; | 
					
						
							|  |  |  | 				} catch (SocketException se) { | 
					
						
							|  |  |  | 					if (se.SocketErrorCode == SocketError.AddressAlreadyInUse) | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 					throw; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			throw new Exception ("Unable to create HttpListener after 10 attempts"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Creates and starts an HttpListener using the specified host, path | 
					
						
							|  |  |  | 		// and authSchemes. The method will try to find an unused port, and | 
					
						
							|  |  |  | 		// use that (multiple attempts with random port numbers will be made). | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// If specified, the initializer will be called immediately after the | 
					
						
							|  |  |  | 		// HttpListener is created (typical usage would be to set/change | 
					
						
							|  |  |  | 		// properties before starting the listener). Be aware that the | 
					
						
							|  |  |  | 		// initializer can be called multiple times (in case multiple creation | 
					
						
							|  |  |  | 		// attempts have to be made). | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// The resulting uri will also be returned (this is just host + port + path). | 
					
						
							|  |  |  | 		public static HttpListener CreateAndStartHttpListener (string host, out int port, string path, out string uri, AuthenticationSchemes? authSchemes = null, Action<HttpListener> initializer = null) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			var rv = CreateAndStartHttpListener (host, out port, path, authSchemes, initializer); | 
					
						
							|  |  |  | 			uri = host + port + path; | 
					
						
							|  |  |  | 			return rv; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |