/metastudio/src/org/meta/net/IPExpression.java

http://metastudio.googlecode.com/ · Java · 283 lines · 154 code · 37 blank · 92 comment · 35 complexity · 06092684a05a0f5b3b6381feb40074c2 MD5 · raw file

  1. /*
  2. * IPExpression.java
  3. *
  4. * Created on July 16, 2006, 7:21 AM
  5. */
  6. package org.meta.net;
  7. import java.net.InetAddress;
  8. import java.util.ArrayList;
  9. import java.util.Iterator;
  10. /**
  11. * Understands IP regular expression, say A:B:C:* and provides a mechanism
  12. * to iterate through all the IP addresses satisfying the wild-card notation
  13. * of the IP. Also provides a way to check if a arbitary IP satisfies
  14. * the IP regular expression rule.
  15. * <pre>
  16. * Limitations:
  17. * a) In its current form, the algorithm only works for a local
  18. * IPV4 based network.
  19. * b) You can only provide one wild-card search. Two stars are not allowed.
  20. * eg. A:B:*:* is not allowed.
  21. * c) However, you may provide a part of the IP with two '?'s as in
  22. * A:B:C:1??, but A:2?:C:1? is illegal.
  23. * d) You can not include both '?' and '*' at the same time in the ip
  24. * expression
  25. * </pre>
  26. *
  27. * Note: You need to provide IP addresses with ':' instead of '.' (IPV4).
  28. * IPV6 is not currently accepted. (but the algo. will try to convert...)
  29. *
  30. * @author V.Ganesh
  31. * @version 2.0 (Part of MeTA v2.0)
  32. */
  33. public class IPExpression {
  34. private boolean isLoacalAddress;
  35. private String [] words;
  36. private int wildCardIndex;
  37. private ArrayList<InetAddress> ipList;
  38. private Iterator<InetAddress> ips;
  39. /** Creates a new instance of IPExpression
  40. *
  41. * @throws UnsupportedOperationException if invalid expression
  42. */
  43. public IPExpression(String ipExpression) {
  44. this.ipExpression = ipExpression;
  45. checkExpression();
  46. generateAddress();
  47. }
  48. /**
  49. * Check if this IP address satisfies the IP expression?
  50. *
  51. * @return a boolean true / false indicating if the IP address matches the
  52. * criterion specified in the IP expression.
  53. */
  54. public boolean matches(InetAddress ip) {
  55. return ipList.contains(ip);
  56. }
  57. /**
  58. * Generate IP addresses
  59. */
  60. private void generateAddress() {
  61. ipList = new ArrayList<InetAddress>();
  62. if (isLoacalAddress) {
  63. ips = ipList.iterator();
  64. return;
  65. } // end if
  66. // next depending on * of ? as wild-cards we generate the IP address
  67. // and search for MeTA services at these addresses
  68. if (words[wildCardIndex].indexOf('*') >= 0)
  69. multiCardIP(words, wildCardIndex);
  70. else if (words[wildCardIndex].indexOf('?') >= 0)
  71. singleCardIP(words, wildCardIndex);
  72. else {
  73. try {
  74. // try to siliently convert all ':' to '.'s
  75. ipExpression = ipExpression.replace(':', '.');
  76. InetAddress newAddress = InetAddress.getByName(ipExpression);
  77. ipList.add(newAddress);
  78. } catch (Exception e) {
  79. System.err.println("Unexpected error in " +
  80. "IPExpression.generateAddress(): " + e);
  81. e.printStackTrace();
  82. } // end of try .. catch block
  83. } // end if
  84. ips = ipList.iterator();
  85. }
  86. /**
  87. * '*' based IP search
  88. *
  89. * @param words components of IP address
  90. * @param wildCardIndex the index in the component containing the wild-card
  91. */
  92. private void multiCardIP(String [] words, int wildCardIndex) {
  93. for(int i=1; i<=Byte.MAX_VALUE*2; i++) {
  94. words[wildCardIndex] = "" + i;
  95. String addr = "";
  96. for(int j=0; j<words.length-1; j++) addr += words[j]+".";
  97. addr += words[words.length-1];
  98. try {
  99. InetAddress newAddress = InetAddress.getByName(addr);
  100. ipList.add(newAddress);
  101. } catch (Exception e) {
  102. System.err.println("Unexpected error in " +
  103. "IPExpression.generateAddress(): " + e);
  104. e.printStackTrace();
  105. continue;
  106. } // end of try .. catch block
  107. } // end for
  108. }
  109. /**
  110. * '?' based IP search
  111. *
  112. * @param words components of IP address
  113. * @param wildCardIndex the index in the component containing the wild-card
  114. */
  115. private void singleCardIP(String [] words, int wildCardIndex) {
  116. // count number of '?'s, if more than three, then do a full search
  117. int qCount = 0;
  118. for(int i=0; i<words[wildCardIndex].length(); i++) {
  119. if (words[wildCardIndex].charAt(i) == '?') qCount++;
  120. } // end for
  121. if (qCount >= 3) {
  122. multiCardIP(words, wildCardIndex);
  123. return;
  124. } // end if
  125. int indx = words[wildCardIndex].indexOf('?');
  126. int lastIndx = words[wildCardIndex].lastIndexOf('?');
  127. if (indx == lastIndx) singleCardIPSub(words, wildCardIndex, indx);
  128. else {
  129. for(int i=0; i<10; i++) {
  130. words[wildCardIndex] = words[wildCardIndex].substring(0, indx)
  131. + i + words[wildCardIndex].substring(indx+1,
  132. words[wildCardIndex].length());
  133. singleCardIPSub(words, wildCardIndex, lastIndx);
  134. } // end if
  135. } // end if
  136. }
  137. /**
  138. * Helper for singleCardIP()
  139. *
  140. * @param words components of IP address
  141. * @param wildCardIndex the index in the component containing the wild-card
  142. * @param indx index of '?' under consideration
  143. */
  144. private void singleCardIPSub(String [] words, int wildCardIndex, int indx) {
  145. for(int i=0; i<10; i++) {
  146. words[wildCardIndex] = words[wildCardIndex].substring(0, indx) + i +
  147. words[wildCardIndex].substring(indx+1,
  148. words[wildCardIndex].length());
  149. if (Integer.parseInt(words[wildCardIndex]) > 254) break;
  150. String addr = "";
  151. for(int j=0; j<words.length-1; j++) addr += words[j]+".";
  152. addr += words[words.length-1];
  153. try {
  154. InetAddress newAddress = InetAddress.getByName(addr);
  155. ipList.add(newAddress);
  156. } catch (Exception e) {
  157. System.err.println("Unexpected error in " +
  158. "IPExpression.generateAddress(): " + e);
  159. e.printStackTrace();
  160. continue;
  161. } // end of try .. catch block
  162. } // end for
  163. }
  164. /**
  165. * check the IP expression
  166. *
  167. * @throws UnsupportedOperationException if invalid expression
  168. */
  169. private void checkExpression() {
  170. // try to siliently convert all dots to ':'
  171. ipExpression = ipExpression.replace('.', ':');
  172. // check the expression for any violations
  173. // first check for double '*' s
  174. if (ipExpression.indexOf('*') != ipExpression.lastIndexOf('*')) {
  175. throw new UnsupportedOperationException("Illegal expression, " +
  176. " * can be used only once in expression: " + ipExpression);
  177. } // end if
  178. // then check if '*' and '?' occur together
  179. if ((ipExpression.indexOf('*') >= 0)
  180. && (ipExpression.indexOf('?') >= 0)) {
  181. throw new UnsupportedOperationException("Illegal expression, " +
  182. " * and ? can not be used together in expression: "
  183. + ipExpression);
  184. } // end if
  185. // next check if more than one '?'s are present
  186. // if so, all of them should be part of only one part of
  187. // the dotted notation:
  188. // A:B:C:1?? is valid, so is A:2?:C:D but not A:2?:C:1?
  189. if (ipExpression.indexOf('?') != ipExpression.lastIndexOf('?')) {
  190. String [] words = ipExpression.split(":");
  191. int qFound = 0;
  192. for(int i=0; i<words.length; i++) {
  193. if (words[i].indexOf('?') >= 0) qFound++;
  194. } // end for
  195. if (qFound > 1) {
  196. throw new UnsupportedOperationException("Illegal expression, " +
  197. " ? can be used in only one sub-part of the expression: "
  198. + ipExpression);
  199. } // end if
  200. } // end if
  201. // if we reach here, probably the expession is a valid one
  202. // but still needs to be checked if each part is a number
  203. words = ipExpression.split(":");
  204. wildCardIndex = 0;
  205. for(int i=0; i<words.length; i++) {
  206. if ((words[i].indexOf('?') >= 0) || ((words[i].indexOf('*') >= 0))){
  207. wildCardIndex = i;
  208. continue;
  209. } // end if
  210. try {
  211. // check if it is an integer?
  212. Integer.parseInt(words[i]);
  213. if (i==0) {
  214. // check if a local address is given for probing
  215. // if so we return back!!
  216. if (Integer.parseInt(words[i]) == 127) {
  217. isLoacalAddress = true;
  218. return;
  219. } // end if
  220. } // end if
  221. } catch (Exception e) {
  222. throw new UnsupportedOperationException("Illegal expression, " +
  223. " sub-parts of expression not containing ? and * should " +
  224. "be integers: " + ipExpression);
  225. } // end of try .. catch block
  226. } // end for
  227. }
  228. /**
  229. * Holds value of property ipExpression.
  230. */
  231. private String ipExpression;
  232. /**
  233. * Getter for property ipExpression.
  234. * @return Value of property ipExpression.
  235. */
  236. public String getIpExpression() {
  237. return this.ipExpression;
  238. }
  239. /**
  240. * Return the next IP address in the list, or a null if none is available.
  241. */
  242. public InetAddress getNextAddress() {
  243. if (ips.hasNext()) return ips.next();
  244. else return null;
  245. }
  246. } // end of class IPExpression