/tags/1.0.0/java/kouchat/src/net/usikkert/kouchat/net/MessageReceiver.java

http://kouchat.googlecode.com/ · Java · 258 lines · 144 code · 39 blank · 75 comment · 20 complexity · d814ab7ff5bab0238f7984bd3ed53059 MD5 · raw file

  1. /***************************************************************************
  2. * Copyright 2006-2009 by Christian Ihle *
  3. * kontakt@usikkert.net *
  4. * *
  5. * This program is free software; you can redistribute it and/or modify *
  6. * it under the terms of the GNU General Public License as published by *
  7. * the Free Software Foundation; either version 2 of the License, or *
  8. * (at your option) any later version. *
  9. * *
  10. * This program is distributed in the hope that it will be useful, *
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of *
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
  13. * GNU General Public License for more details. *
  14. * *
  15. * You should have received a copy of the GNU General Public License *
  16. * along with this program; if not, write to the *
  17. * Free Software Foundation, Inc., *
  18. * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
  19. ***************************************************************************/
  20. package net.usikkert.kouchat.net;
  21. import java.io.IOException;
  22. import java.net.DatagramPacket;
  23. import java.net.InetAddress;
  24. import java.net.MulticastSocket;
  25. import java.net.NetworkInterface;
  26. import java.util.logging.Level;
  27. import java.util.logging.Logger;
  28. import net.usikkert.kouchat.Constants;
  29. import net.usikkert.kouchat.event.ReceiverListener;
  30. import net.usikkert.kouchat.misc.ErrorHandler;
  31. /**
  32. * This is the thread that listens for multicast messages from
  33. * the network, and notifies any listeners when messages arrive.
  34. *
  35. * @author Christian Ihle
  36. */
  37. public class MessageReceiver implements Runnable
  38. {
  39. /** The logger. */
  40. private static final Logger LOG = Logger.getLogger( MessageReceiver.class.getName() );
  41. /** The multicast socket used for receiving messages. */
  42. private MulticastSocket mcSocket;
  43. /** The inetaddress object with the multicast ip address to receive messages from. */
  44. private InetAddress address;
  45. /** The listener getting all the messages received here. */
  46. private ReceiverListener listener;
  47. /** If connected to the network or not. */
  48. private boolean connected;
  49. /** The background thread watching for messages from the network. */
  50. private Thread worker;
  51. /** The error handler for registering important messages. */
  52. private final ErrorHandler errorHandler;
  53. /** The port to receive messages on. */
  54. private final int port;
  55. /**
  56. * Default constructor.
  57. *
  58. * <p>Initializes the network with the default ip address and port.</p>
  59. *
  60. * @see Constants#NETWORK_IP
  61. * @see Constants#NETWORK_CHAT_PORT
  62. */
  63. public MessageReceiver()
  64. {
  65. this( Constants.NETWORK_IP, Constants.NETWORK_CHAT_PORT );
  66. }
  67. /**
  68. * Alternative constructor.
  69. *
  70. * <p>Initializes the network with the given ip address and port.</p>
  71. *
  72. * @param ipAddress Multicast ip address to connect to.
  73. * @param port Port to connect to.
  74. */
  75. public MessageReceiver( final String ipAddress, final int port )
  76. {
  77. this.port = port;
  78. errorHandler = ErrorHandler.getErrorHandler();
  79. try
  80. {
  81. address = InetAddress.getByName( ipAddress );
  82. }
  83. catch ( final IOException e )
  84. {
  85. LOG.log( Level.SEVERE, e.toString(), e );
  86. errorHandler.showCriticalError( "Failed to initialize the network:\n" + e + "\n"
  87. + Constants.APP_NAME + " will now shutdown." );
  88. System.exit( 1 );
  89. }
  90. }
  91. /**
  92. * Waits for incoming packets, and notifies the listener when they arrive.
  93. */
  94. public void run()
  95. {
  96. while ( connected )
  97. {
  98. try
  99. {
  100. DatagramPacket packet = new DatagramPacket(
  101. new byte[Constants.NETWORK_PACKET_SIZE], Constants.NETWORK_PACKET_SIZE );
  102. if ( connected )
  103. {
  104. mcSocket.receive( packet );
  105. String ip = packet.getAddress().getHostAddress();
  106. String message = new String( packet.getData(), Constants.MESSAGE_CHARSET ).trim();
  107. LOG.log( Level.FINE, "Message arrived from " + ip + ": " + message );
  108. if ( listener != null )
  109. listener.messageArrived( message, ip );
  110. }
  111. }
  112. // Happens when socket is closed, or network is down
  113. catch ( final IOException e )
  114. {
  115. if ( connected )
  116. LOG.log( Level.WARNING, e.toString() );
  117. else
  118. LOG.log( Level.FINE, e.toString() );
  119. }
  120. }
  121. }
  122. /**
  123. * Starts the thread that listens for messages.
  124. */
  125. private void startThread()
  126. {
  127. LOG.log( Level.FINE, "Starting." );
  128. worker = new Thread( this, "MessageReceiverWorker" );
  129. worker.start();
  130. }
  131. /**
  132. * Connects to the network with the given network interface, or gives
  133. * the control to the operating system to choose if <code>null</code>
  134. * is given.
  135. *
  136. * <p>Will also start a thread to continuously receive messages.</p>
  137. *
  138. * @param networkInterface The network interface to use, or <code>null</code>.
  139. * @return If connected to the network or not.
  140. */
  141. public synchronized boolean startReceiver( final NetworkInterface networkInterface )
  142. {
  143. LOG.log( Level.FINE, "Connecting..." );
  144. try
  145. {
  146. if ( connected )
  147. {
  148. LOG.log( Level.FINE, "Already connected." );
  149. }
  150. else
  151. {
  152. if ( mcSocket == null )
  153. mcSocket = new MulticastSocket( port );
  154. if ( networkInterface != null )
  155. mcSocket.setNetworkInterface( networkInterface );
  156. mcSocket.joinGroup( address );
  157. LOG.log( Level.FINE, "Connected to " + mcSocket.getNetworkInterface().getDisplayName() + "." );
  158. connected = true;
  159. }
  160. }
  161. catch ( final IOException e )
  162. {
  163. LOG.log( Level.SEVERE, "Could not start receiver: " + e.toString() );
  164. if ( mcSocket != null )
  165. {
  166. if ( !mcSocket.isClosed() )
  167. mcSocket.close();
  168. mcSocket = null;
  169. }
  170. }
  171. if ( connected && ( worker == null || !worker.isAlive() ) )
  172. {
  173. startThread();
  174. }
  175. return connected;
  176. }
  177. /**
  178. * Disconnects from the network and closes the multicast socket.
  179. */
  180. public synchronized void stopReceiver()
  181. {
  182. LOG.log( Level.FINE, "Disconnecting..." );
  183. if ( !connected )
  184. {
  185. LOG.log( Level.FINE, "Not connected." );
  186. }
  187. else
  188. {
  189. connected = false;
  190. try
  191. {
  192. if ( !mcSocket.isClosed() )
  193. {
  194. mcSocket.leaveGroup( address );
  195. }
  196. }
  197. catch ( final IOException e )
  198. {
  199. LOG.log( Level.WARNING, e.toString() );
  200. }
  201. if ( !mcSocket.isClosed() )
  202. {
  203. mcSocket.close();
  204. mcSocket = null;
  205. }
  206. LOG.log( Level.FINE, "Disconnected." );
  207. }
  208. }
  209. /**
  210. * Registers as the listener to receive all the messages from
  211. * the network.
  212. *
  213. * @param listener The listener to register.
  214. */
  215. public void registerReceiverListener( final ReceiverListener listener )
  216. {
  217. this.listener = listener;
  218. }
  219. }