/trunk/tigermud/TigerLIB/CommsLib/TcpComms/TcpConnectionListener.cs

# · C# · 278 lines · 178 code · 20 blank · 80 comment · 6 complexity · d8c0962c035fab4ab0ba67ef18becac8 MD5 · raw file

  1. #region TigerMUD License
  2. /*
  3. /-------------------------------------------------------------\
  4. | _______ _ __ __ _ _ _____ |
  5. | |__ __|(_) | \/ || | | || __ \ |
  6. | | | _ __ _ ___ _ __ | \ / || | | || | | | |
  7. | | | | | / _` | / _ \| '__|| |\/| || | | || | | | |
  8. | | | | || (_| || __/| | | | | || |__| || |__| | |
  9. | |_| |_| \__, | \___||_| |_| |_| \____/ |_____/ |
  10. | __/ | |
  11. | |___/ Copyright (c) 2004 |
  12. \-------------------------------------------------------------/
  13. TigerMUD. A Multi User Dungeon engine.
  14. Copyright (C) 2004 Adam Miller et al.
  15. This program is free software; you can redistribute it and/or modify
  16. it under the terms of the GNU General Public License as published by
  17. the Free Software Foundation; either version 2 of the License, or
  18. (at your option) any later version.
  19. This program is distributed in the hope that it will be useful,
  20. but WITHOUT ANY WARRANTY; without even the implied warranty of
  21. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  22. GNU General Public License for more details.
  23. You should have received a copy of the GNU General Public License
  24. along with this program; if not, write to the Free Software
  25. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  26. You can contact the TigerMUD developers at www.tigermud.com or at
  27. http://sourceforge.net/projects/tigermud.
  28. The full licence can be found in <root>/docs/TigerMUD_license.txt
  29. */
  30. #endregion
  31. using System;
  32. using System.Net;
  33. using System.Net.Sockets;
  34. using System.IO;
  35. using System.Threading;
  36. using System.Collections;
  37. using TigerMUD.CommsLib;
  38. namespace TigerMUD.CommsLib.TcpComms
  39. {
  40. /// <summary>
  41. /// Listens out for new connections from users.
  42. /// </summary>
  43. public class TcpConnectionListener : IConnectionListener
  44. {
  45. private TcpListener tcpListener;
  46. private bool enabled;
  47. private ArrayList connectedClients;
  48. private ReaderWriterLock clientsLock;
  49. private Thread listenerThread;
  50. private bool acceptNewConnections;
  51. public TcpConnectionListener(int port)
  52. {
  53. tcpListener = new TcpListener(IPAddress.Any, port);
  54. listenerThread = new Thread(new ThreadStart(ListenForConnections));
  55. enabled = false;
  56. connectedClients = new ArrayList();
  57. clientsLock = new ReaderWriterLock();
  58. acceptNewConnections = true;
  59. }
  60. #region Public Properties
  61. /// <summary>
  62. /// Gets the status of the client listener.
  63. /// </summary>
  64. public bool Active
  65. {
  66. get
  67. {
  68. return enabled;
  69. }
  70. }
  71. /// <summary>
  72. /// Returns the number of connected clients.
  73. /// </summary>
  74. public int ClientCount
  75. {
  76. get
  77. {
  78. clientsLock.AcquireReaderLock(Timeout.Infinite);
  79. try
  80. {
  81. return connectedClients.Count;
  82. }
  83. finally
  84. {
  85. clientsLock.ReleaseLock();
  86. }
  87. }
  88. }
  89. public bool AcceptNewConnections
  90. {
  91. get
  92. {
  93. return acceptNewConnections;
  94. }
  95. set
  96. {
  97. acceptNewConnections = value;
  98. }
  99. }
  100. #endregion
  101. public void Start()
  102. {
  103. enabled = true;
  104. listenerThread.Start();
  105. }
  106. private void ListenForConnections()
  107. {
  108. try
  109. {
  110. // Start the listener listening for requests.
  111. tcpListener.Start();
  112. while (enabled)
  113. {
  114. while (!tcpListener.Pending() && enabled)
  115. {
  116. Thread.Sleep(500);
  117. }
  118. // Make sure that while we were sleeping
  119. // the service wasn't enabled.
  120. // Also make sure that we are go for accepting
  121. // new connections.
  122. if (enabled)
  123. {
  124. // Get the socket.
  125. TcpClientSocket socket = new TcpClientSocket(tcpListener.AcceptSocket());
  126. if (acceptNewConnections)
  127. {
  128. // Create a new Client Listener and pass through the callback
  129. ClientListener tcp = new ClientListener(socket,
  130. new ClientDisconnectedEventHandler(this.ClientDisconnectedCallback));
  131. // Add the client listener to the collection
  132. clientsLock.AcquireWriterLock(Timeout.Infinite);
  133. try
  134. {
  135. connectedClients.Add(tcp);
  136. }
  137. finally
  138. {
  139. clientsLock.ReleaseLock();
  140. }
  141. // Start the client listener.
  142. tcp.StartListening();
  143. }
  144. else
  145. {
  146. // We simply close connections if we
  147. // are not accepting connections.
  148. socket.Close();
  149. }
  150. // Sleep the current thread for a bit.
  151. Thread.Sleep(1);
  152. }
  153. }
  154. }
  155. catch (Exception ex)
  156. {
  157. this.enabled = false;
  158. Lib.log.Add("TcpConnectionListener.ListenForConnections exception", ex.Message + ex.StackTrace);
  159. }
  160. finally
  161. {
  162. tcpListener.Stop();
  163. }
  164. }
  165. /// <summary>
  166. /// Stop the connection listener.
  167. /// </summary>
  168. public void Stop()
  169. {
  170. enabled = false;
  171. // Stop all the client listener threads.
  172. StopClientListeners();
  173. }
  174. /// <summary>
  175. /// Stops all the client listener threads.
  176. /// </summary>
  177. private void StopClientListeners()
  178. {
  179. // 1. First we try requesting them to stop on their own.
  180. clientsLock.AcquireReaderLock(Timeout.Infinite);
  181. try
  182. {
  183. foreach (ClientListener clientListener in connectedClients)
  184. {
  185. clientListener.StopListening();
  186. }
  187. }
  188. finally
  189. {
  190. clientsLock.ReleaseLock();
  191. }
  192. // 2. Next we sleep for a second to give the listeners
  193. // a chance to disconnect and remove themselves from
  194. // the connectedClients collection.
  195. System.Threading.Thread.Sleep(1000);
  196. // 3. If there are still connected users, we Lib.Disco
  197. // the users and abort their threads.
  198. clientsLock.AcquireReaderLock(Timeout.Infinite);
  199. try
  200. {
  201. foreach (ClientListener cl in connectedClients)
  202. {
  203. // Try to save his data
  204. Lib.Disco(cl.UserSocket, "server shutdown");
  205. // Abort his user thread.
  206. cl.ListenerThread.Abort();
  207. }
  208. }
  209. finally
  210. {
  211. clientsLock.ReleaseLock();
  212. }
  213. // Go through and see if there are any connected clients
  214. clientsLock.AcquireReaderLock(Timeout.Infinite);
  215. try
  216. {
  217. foreach (ClientListener cl in connectedClients)
  218. {
  219. if (cl.Active)
  220. {
  221. // Abort the thread again
  222. cl.ListenerThread.Abort();
  223. }
  224. }
  225. }
  226. finally
  227. {
  228. clientsLock.ReleaseLock();
  229. }
  230. // There are no more connected clients.
  231. }
  232. /// <summary>
  233. /// Called when the ClientListener has quit.
  234. /// </summary>
  235. /// <param name="sender">The client listener that quit.</param>
  236. private void ClientDisconnectedCallback(ClientListener sender)
  237. {
  238. clientsLock.AcquireWriterLock(Timeout.Infinite);
  239. try
  240. {
  241. // Lib.Disco this connection just to be safe
  242. //Lib.Disco(sender.UserSocket);
  243. // Remove this client listener from the collection
  244. connectedClients.Remove(sender);
  245. }
  246. finally
  247. {
  248. clientsLock.ReleaseLock();
  249. }
  250. }
  251. }
  252. }