/hadoop-common-project/hadoop-common/src/main/java/org/apache/hadoop/util/MachineList.java

http://github.com/apache/hadoop-common · Java · 210 lines · 130 code · 20 blank · 60 comment · 36 complexity · 6fab22ab35f1038a40f439e500e15d49 MD5 · raw file

  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. package org.apache.hadoop.util;
  19. import java.net.InetAddress;
  20. import java.net.UnknownHostException;
  21. import java.util.ArrayList;
  22. import java.util.Collection;
  23. import java.util.HashSet;
  24. import java.util.LinkedList;
  25. import java.util.List;
  26. import java.util.Set;
  27. import org.apache.commons.logging.Log;
  28. import org.apache.commons.logging.LogFactory;
  29. import org.apache.commons.net.util.SubnetUtils;
  30. import com.google.common.annotations.VisibleForTesting;
  31. import com.google.common.net.InetAddresses;
  32. /**
  33. * Container class which holds a list of ip/host addresses and
  34. * answers membership queries.
  35. *
  36. * Accepts list of ip addresses, ip addreses in CIDR format and/or
  37. * host addresses.
  38. */
  39. public class MachineList {
  40. public static final Log LOG = LogFactory.getLog(MachineList.class);
  41. /**
  42. * InetAddressFactory is used to obtain InetAddress from host.
  43. * This class makes it easy to simulate host to ip mappings during testing.
  44. *
  45. */
  46. public static class InetAddressFactory {
  47. static final InetAddressFactory S_INSTANCE = new InetAddressFactory();
  48. public InetAddress getByName (String host) throws UnknownHostException {
  49. return InetAddress.getByName(host);
  50. }
  51. }
  52. private final boolean all;
  53. private final Set<String> ipAddresses;
  54. private final List<SubnetUtils.SubnetInfo> cidrAddresses;
  55. private final Set<String> hostNames;
  56. private final InetAddressFactory addressFactory;
  57. /**
  58. *
  59. * @param hostEntries comma separated ip/cidr/host addresses
  60. */
  61. public MachineList(String hostEntries) {
  62. this(StringUtils.getTrimmedStringCollection(hostEntries));
  63. }
  64. /**
  65. *
  66. * @param hostEntries collection of separated ip/cidr/host addresses
  67. */
  68. public MachineList(Collection<String> hostEntries) {
  69. this(hostEntries, InetAddressFactory.S_INSTANCE);
  70. }
  71. /**
  72. * Accepts a collection of ip/cidr/host addresses
  73. *
  74. * @param hostEntries
  75. * @param addressFactory addressFactory to convert host to InetAddress
  76. */
  77. public MachineList(Collection<String> hostEntries, InetAddressFactory addressFactory) {
  78. this.addressFactory = addressFactory;
  79. if (hostEntries != null) {
  80. if ((hostEntries.size() == 1) && (hostEntries.contains("*"))) {
  81. all = true;
  82. ipAddresses = null;
  83. hostNames = null;
  84. cidrAddresses = null;
  85. } else {
  86. all = false;
  87. Set<String> ips = new HashSet<String>();
  88. List<SubnetUtils.SubnetInfo> cidrs = new LinkedList<SubnetUtils.SubnetInfo>();
  89. Set<String> hosts = new HashSet<String>();
  90. for (String hostEntry : hostEntries) {
  91. //ip address range
  92. if (hostEntry.indexOf("/") > -1) {
  93. try {
  94. SubnetUtils subnet = new SubnetUtils(hostEntry);
  95. subnet.setInclusiveHostCount(true);
  96. cidrs.add(subnet.getInfo());
  97. } catch (IllegalArgumentException e) {
  98. LOG.warn("Invalid CIDR syntax : " + hostEntry);
  99. throw e;
  100. }
  101. } else if (InetAddresses.isInetAddress(hostEntry)) { //ip address
  102. ips.add(hostEntry);
  103. } else { //hostname
  104. hosts.add(hostEntry);
  105. }
  106. }
  107. ipAddresses = (ips.size() > 0) ? ips : null;
  108. cidrAddresses = (cidrs.size() > 0) ? cidrs : null;
  109. hostNames = (hosts.size() > 0) ? hosts : null;
  110. }
  111. } else {
  112. all = false;
  113. ipAddresses = null;
  114. hostNames = null;
  115. cidrAddresses = null;
  116. }
  117. }
  118. /**
  119. * Accepts an ip address and return true if ipAddress is in the list
  120. * @param ipAddress
  121. * @return true if ipAddress is part of the list
  122. */
  123. public boolean includes(String ipAddress) {
  124. if (all) {
  125. return true;
  126. }
  127. //check in the set of ipAddresses
  128. if ((ipAddresses != null) && ipAddresses.contains(ipAddress)) {
  129. return true;
  130. }
  131. //iterate through the ip ranges for inclusion
  132. if (cidrAddresses != null) {
  133. for(SubnetUtils.SubnetInfo cidrAddress : cidrAddresses) {
  134. if(cidrAddress.isInRange(ipAddress)) {
  135. return true;
  136. }
  137. }
  138. }
  139. //check if the ipAddress matches one of hostnames
  140. if (hostNames != null) {
  141. //convert given ipAddress to hostname and look for a match
  142. InetAddress hostAddr;
  143. try {
  144. hostAddr = addressFactory.getByName(ipAddress);
  145. if ((hostAddr != null) && hostNames.contains(hostAddr.getCanonicalHostName())) {
  146. return true;
  147. }
  148. } catch (UnknownHostException e) {
  149. //ignore the exception and proceed to resolve the list of hosts
  150. }
  151. //loop through host addresses and convert them to ip and look for a match
  152. for (String host : hostNames) {
  153. try {
  154. hostAddr = addressFactory.getByName(host);
  155. } catch (UnknownHostException e) {
  156. continue;
  157. }
  158. if (hostAddr.getHostAddress().equals(ipAddress)) {
  159. return true;
  160. }
  161. }
  162. }
  163. return false;
  164. }
  165. /**
  166. * returns the contents of the MachineList as a Collection<String>
  167. * This can be used for testing
  168. * @return contents of the MachineList
  169. */
  170. @VisibleForTesting
  171. public Collection<String> getCollection() {
  172. Collection<String> list = new ArrayList<String>();
  173. if (all) {
  174. list.add("*");
  175. } else {
  176. if (ipAddresses != null) {
  177. list.addAll(ipAddresses);
  178. }
  179. if (hostNames != null) {
  180. list.addAll(hostNames);
  181. }
  182. if (cidrAddresses != null) {
  183. for(SubnetUtils.SubnetInfo cidrAddress : cidrAddresses) {
  184. list.add(cidrAddress.getCidrSignature());
  185. }
  186. }
  187. }
  188. return list;
  189. }
  190. }