/tags/0.9.9/java/kouchat/src/net/usikkert/kouchat/net/NetworkUtils.java

http://kouchat.googlecode.com/ · Java · 304 lines · 155 code · 42 blank · 107 comment · 40 complexity · 305d24743c3073f99c7af4f57d377bcc MD5 · raw file

  1. /***************************************************************************
  2. * Copyright 2006-2008 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.net.Inet4Address;
  22. import java.net.InetAddress;
  23. import java.net.NetworkInterface;
  24. import java.net.SocketException;
  25. import java.util.Enumeration;
  26. import java.util.logging.Level;
  27. import java.util.logging.Logger;
  28. import net.usikkert.kouchat.util.Loggers;
  29. /**
  30. * Class containing utility methods for network operations.
  31. *
  32. * @author Christian Ihle
  33. */
  34. public final class NetworkUtils
  35. {
  36. /** The logger. */
  37. private static final Logger LOG = Loggers.NETWORK_LOG;
  38. /**
  39. * Private constructor. Only static methods here.
  40. */
  41. private NetworkUtils()
  42. {
  43. }
  44. /**
  45. * Checks if the network interface is up, and usable.
  46. *
  47. * <p>A network interface is usable when it:</p>
  48. *
  49. * <ul>
  50. * <li>Is up.</li>
  51. * <li>Supports multicast.</li>
  52. * <li>Is not a loopback device, like localhost.</li>
  53. * <li>Is not a point to point device, like a modem.</li>
  54. * <li>Is not virtual, like <code>eth0:1</code>.</li>
  55. * <li>Is not a virtual machine network interface (vmnet).</li>
  56. * <li>Has an IPv4 address.</li>
  57. * </ul>
  58. *
  59. * @param netif The network interface to check.
  60. * @return True if the network interface is usable.
  61. */
  62. public static boolean isUsable( final NetworkInterface netif )
  63. {
  64. if ( netif == null )
  65. return false;
  66. try
  67. {
  68. return netif.isUp() && !netif.isLoopback() && !netif.isPointToPoint()
  69. && !netif.isVirtual() && netif.supportsMulticast()
  70. && !netif.getName().toLowerCase().contains( "vmnet" )
  71. && !netif.getDisplayName().toLowerCase().contains( "vmnet" )
  72. && hasIPv4Address( netif );
  73. }
  74. catch ( final SocketException e )
  75. {
  76. LOG.log( Level.WARNING, e.toString() );
  77. return false;
  78. }
  79. }
  80. /**
  81. * Checks if the network interface has an IPv4-address.
  82. *
  83. * @param netif The network interface to check.
  84. * @return If an IPv4-address was found or not.
  85. */
  86. public static boolean hasIPv4Address( final NetworkInterface netif )
  87. {
  88. if ( netif == null )
  89. return false;
  90. Enumeration<InetAddress> inetAddresses = netif.getInetAddresses();
  91. while ( inetAddresses.hasMoreElements() )
  92. {
  93. InetAddress inetAddress = inetAddresses.nextElement();
  94. if ( inetAddress instanceof Inet4Address )
  95. return true;
  96. }
  97. return false;
  98. }
  99. /**
  100. * Constructs a string with the information found on a {@link NetworkInterface}.
  101. *
  102. * @param netif The network interface to check.
  103. * @return A string with information.
  104. */
  105. public static String getNetworkInterfaceInfo( final NetworkInterface netif )
  106. {
  107. if ( netif == null )
  108. return "Invalid network interface.";
  109. try
  110. {
  111. return "Interface name: " + netif.getDisplayName() + "\n"
  112. + "Device: " + netif.getName() + "\n"
  113. + "Is loopback: " + netif.isLoopback() + "\n"
  114. + "Is up: " + netif.isUp() + "\n"
  115. + "Is p2p: " + netif.isPointToPoint() + "\n"
  116. + "Is virtual: " + netif.isVirtual() + "\n"
  117. + "Supports multicast: " + netif.supportsMulticast() + "\n"
  118. + "MAC address: " + getMacAddress( netif ) + "\n"
  119. + "IP addresses: " + getIPv4Addresses( netif );
  120. }
  121. catch ( final SocketException e )
  122. {
  123. LOG.log( Level.WARNING, e.toString() );
  124. return "Failed to get network interface information.";
  125. }
  126. }
  127. /**
  128. * Returns a list of the IPv4-addresses on the network interface in string format.
  129. *
  130. * @param netif The network interface to get the IPv4-addresses from.
  131. * @return All the IPv4-addresses on the network interface.
  132. */
  133. public static String getIPv4Addresses( final NetworkInterface netif )
  134. {
  135. if ( netif == null )
  136. return "";
  137. String ipAddress = "";
  138. Enumeration<InetAddress> inetAddresses = netif.getInetAddresses();
  139. while ( inetAddresses.hasMoreElements() )
  140. {
  141. InetAddress inetAddress = inetAddresses.nextElement();
  142. if ( inetAddress instanceof Inet4Address )
  143. ipAddress += inetAddress.getHostAddress() + " ";
  144. }
  145. return ipAddress;
  146. }
  147. /**
  148. * Returns the MAC-address of the network interface, in hex format.
  149. *
  150. * @param netif The network interface to get the MAC-address of.
  151. * @return The MAC-address in hex, as a string.
  152. */
  153. public static String getMacAddress( final NetworkInterface netif )
  154. {
  155. if ( netif == null )
  156. return "";
  157. String macAddress = "";
  158. byte[] address = null;
  159. try
  160. {
  161. address = netif.getHardwareAddress();
  162. }
  163. catch ( final SocketException e )
  164. {
  165. LOG.log( Level.WARNING, e.toString() );
  166. }
  167. if ( address != null )
  168. {
  169. // Convert byte array to hex format
  170. for ( int i = 0; i < address.length; i++ )
  171. {
  172. macAddress += String.format( "%02x", address[i] );
  173. if ( i != address.length - 1 )
  174. macAddress += "-";
  175. }
  176. }
  177. return macAddress.toUpperCase();
  178. }
  179. /**
  180. * Fetches all the network interfaces again, and returns the one
  181. * which is the same as the original network interface.
  182. *
  183. * <p>This is useful to make sure the network interface information
  184. * is up to date, like the current ip address.</p>
  185. *
  186. * @param origNetIf The original network interface to compare with.
  187. * @return An updated version of the same network interface,
  188. * or <code>null</code> if not found.
  189. */
  190. public static NetworkInterface getUpdatedNetworkInterface( final NetworkInterface origNetIf )
  191. {
  192. if ( origNetIf == null )
  193. return null;
  194. Enumeration<NetworkInterface> networkInterfaces = getNetworkInterfaces();
  195. if ( networkInterfaces == null )
  196. return null;
  197. while ( networkInterfaces.hasMoreElements() )
  198. {
  199. NetworkInterface netif = networkInterfaces.nextElement();
  200. if ( sameNetworkInterface( origNetIf, netif ) )
  201. return netif;
  202. }
  203. return null;
  204. }
  205. /**
  206. * Compares 2 network interfaces. The only way the 2 network interfaces
  207. * can be considered the same is if they have the exact same name.
  208. *
  209. * <p>If any of the network interfaces are <code>null</code> then they
  210. * are not considered the same.</p>
  211. *
  212. * @param netIf1 The first network interface.
  213. * @param netIf2 The second network interface.
  214. * @return If they are the same or not.
  215. */
  216. public static boolean sameNetworkInterface( final NetworkInterface netIf1, final NetworkInterface netIf2 )
  217. {
  218. if ( netIf1 == null || netIf2 == null )
  219. return false;
  220. return netIf1.getName().equals( netIf2.getName() );
  221. }
  222. /**
  223. * Iterates through a list of available network interfaces, and returns
  224. * the first that is usable. Returns <code>null</code> if no usable
  225. * interface is found.
  226. *
  227. * @return The first usable network interface, or <code>null</code>.
  228. * @see #isUsable(NetworkInterface)
  229. */
  230. public static NetworkInterface findFirstUsableNetworkInterface()
  231. {
  232. Enumeration<NetworkInterface> networkInterfaces = getNetworkInterfaces();
  233. if ( networkInterfaces == null )
  234. return null;
  235. while ( networkInterfaces.hasMoreElements() )
  236. {
  237. NetworkInterface netif = networkInterfaces.nextElement();
  238. if ( isUsable( netif ) )
  239. return netif;
  240. }
  241. return null;
  242. }
  243. /**
  244. * Gets all the available network interfaces. Returns <code>null</code>
  245. * if the operation fails, or no interfaces are available.
  246. *
  247. * @return All network interfaces, or <code>null</code>.
  248. */
  249. public static Enumeration<NetworkInterface> getNetworkInterfaces()
  250. {
  251. Enumeration<NetworkInterface> networkInterfaces = null;
  252. try
  253. {
  254. networkInterfaces = NetworkInterface.getNetworkInterfaces();
  255. }
  256. catch ( final SocketException e )
  257. {
  258. LOG.log( Level.WARNING, e.toString() );
  259. }
  260. return networkInterfaces;
  261. }
  262. }