/vt-password/tags/vt-password-3.1.1/src/main/java/edu/vt/middleware/password/AbstractSequenceRule.java

http://vt-middleware.googlecode.com/ · Java · 346 lines · 174 code · 48 blank · 124 comment · 38 complexity · c2ae81e6bb6ace592e0520ea83b400f3 MD5 · raw file

  1. /*
  2. $Id: AbstractSequenceRule.java 1930 2011-04-28 17:06:17Z dfisher $
  3. Copyright (C) 2003-2011 Virginia Tech.
  4. All rights reserved.
  5. SEE LICENSE FOR MORE INFORMATION
  6. Author: Middleware Services
  7. Email: middleware@vt.edu
  8. Version: $Revision: 1930 $
  9. Updated: $Date: 2011-04-28 19:06:17 +0200 (Thu, 28 Apr 2011) $
  10. */
  11. package edu.vt.middleware.password;
  12. import java.util.LinkedHashMap;
  13. import java.util.Map;
  14. /**
  15. * Provide common implementation for keyboard sequence rules.
  16. *
  17. * @author Middleware Services
  18. * @version $Revision: 1930 $ $Date: 2011-04-28 19:06:17 +0200 (Thu, 28 Apr 2011) $
  19. */
  20. public abstract class AbstractSequenceRule implements Rule
  21. {
  22. /** Error code for sequence validation failures. */
  23. public static final String ERROR_CODE = "ILLEGAL_SEQUENCE";
  24. /** Default length of keyboard sequence, value is {@value}. */
  25. public static final int DEFAULT_SEQUENCE_LENGTH = 5;
  26. /** Minimum length of keyboard sequence, value is {@value}. */
  27. public static final int MINIMUM_SEQUENCE_LENGTH = 3;
  28. /** Number of characters in sequence to match. */
  29. protected int sequenceLength = DEFAULT_SEQUENCE_LENGTH;
  30. /** Whether or not to wrap a sequence when searching for matches. */
  31. protected boolean wrapSequence;
  32. /** {@inheritDoc} */
  33. @Override
  34. public RuleResult validate(final PasswordData passwordData)
  35. {
  36. final RuleResult result = new RuleResult(true);
  37. final String password = passwordData.getPassword().getText();
  38. final int max = password.length() - sequenceLength + 1;
  39. Sequence sequence;
  40. int position;
  41. char c;
  42. for (int i = 0; i < getSequenceCount(); i++) {
  43. for (int j = 0; j < max; j++) {
  44. sequence = newSequence(getSequence(i), password.charAt(j));
  45. if (sequence != null) {
  46. position = j;
  47. while (sequence.forward()) {
  48. c = password.charAt(++position);
  49. if (c == sequence.currentLower() || c == sequence.currentUpper()) {
  50. sequence.addMatchCharacter(c);
  51. } else {
  52. break;
  53. }
  54. }
  55. if (sequence.matchCount() == sequenceLength) {
  56. recordFailure(result, sequence.matchString());
  57. }
  58. sequence.reset();
  59. position = j;
  60. while (sequence.backward()) {
  61. c = password.charAt(++position);
  62. if (c == sequence.currentLower() || c == sequence.currentUpper()) {
  63. sequence.addMatchCharacter(c);
  64. } else {
  65. break;
  66. }
  67. }
  68. if (sequence.matchCount() == sequenceLength) {
  69. recordFailure(result, sequence.matchString());
  70. }
  71. }
  72. }
  73. }
  74. return result;
  75. }
  76. /**
  77. * Returns a string representation of this object.
  78. *
  79. * @return string representation
  80. */
  81. @Override
  82. public String toString()
  83. {
  84. return
  85. String.format(
  86. "%s@%h::length=%d,wrap=%s",
  87. getClass().getName(),
  88. hashCode(),
  89. sequenceLength,
  90. wrapSequence);
  91. }
  92. /**
  93. * Sets the sequence length.
  94. *
  95. * @param sl sequence length
  96. */
  97. protected void setSequenceLength(final int sl)
  98. {
  99. if (sl < MINIMUM_SEQUENCE_LENGTH) {
  100. throw new IllegalArgumentException(
  101. String.format(
  102. "sequence length must be >= %s",
  103. MINIMUM_SEQUENCE_LENGTH));
  104. }
  105. sequenceLength = sl;
  106. }
  107. /**
  108. * Returns the sequence of character pairs for which to search.
  109. *
  110. * @param n provides support for multiple character sequences that are
  111. * indexed from 0 to n.
  112. *
  113. * @return character sequence.
  114. */
  115. protected abstract char[][] getSequence(final int n);
  116. /**
  117. * Returns the number of character sequences used in this implementation.
  118. *
  119. * @return number of character sequences.
  120. */
  121. protected abstract int getSequenceCount();
  122. /**
  123. * Creates an iterator that iterates over a character sequence positioned at
  124. * the first matching character, if any, in the given password.
  125. *
  126. * @param chars sequence of upper/lowercase character pairs.
  127. * @param first first character to match in character sequence.
  128. *
  129. * @return forward sequence iterator.
  130. */
  131. private Sequence newSequence(final char[][] chars, final char first)
  132. {
  133. for (int i = 0; i < chars.length; i++) {
  134. if (first == chars[i][0] || first == chars[i][1]) {
  135. final Sequence s = new Sequence(chars, i, sequenceLength, wrapSequence);
  136. s.addMatchCharacter(first);
  137. return s;
  138. }
  139. }
  140. return null;
  141. }
  142. /**
  143. * Records a validation failure.
  144. *
  145. * @param result rule result holding failure details.
  146. * @param match illegal string matched in the password that caused failure.
  147. */
  148. private void recordFailure(final RuleResult result, final String match)
  149. {
  150. final Map<String, Object> m = new LinkedHashMap<String, Object>();
  151. m.put("sequence", match);
  152. result.setValid(false);
  153. result.getDetails().add(new RuleResultDetail(ERROR_CODE, m));
  154. }
  155. /**
  156. * Contains convenience methods for iterating over a sequence of
  157. * upper/lowercase pairs of characters and stores matched characters.
  158. *
  159. * @author Middleware Services
  160. * @version $Revision: 1930 $
  161. */
  162. private class Sequence
  163. {
  164. /** Sequence of upper/lower character pairs. */
  165. private final char[][] chars;
  166. /** 0-based iterator start position. */
  167. private int start;
  168. /** Number of characters to iterate over. */
  169. private int length;
  170. /** Index upper bound. */
  171. private int ubound;
  172. /** Index lower bound. */
  173. private int lbound;
  174. /** Current 0-based iterator position. */
  175. private int position;
  176. /** Stores matched characters. */
  177. private StringBuilder matches;
  178. /**
  179. * Creates a new sequence.
  180. *
  181. * @param characters sequence of characters
  182. * @param startIndex in the characters array
  183. * @param count length of this sequence
  184. * @param wrap whether this sequence wraps
  185. */
  186. public Sequence(
  187. final char[][] characters,
  188. final int startIndex,
  189. final int count,
  190. final boolean wrap)
  191. {
  192. chars = characters;
  193. start = startIndex;
  194. length = count;
  195. lbound = start - length;
  196. ubound = start + length;
  197. if (lbound < 0 && !wrap) {
  198. lbound = 0;
  199. }
  200. if (ubound >= characters.length && !wrap) {
  201. ubound = characters.length;
  202. }
  203. position = start;
  204. matches = new StringBuilder(length);
  205. }
  206. /**
  207. * Advances the iterator one unit in the forward direction.
  208. *
  209. * @return true if characters remain, false otherwise.
  210. */
  211. public boolean forward()
  212. {
  213. return ++position < ubound;
  214. }
  215. /**
  216. * Advances the iterator one unit in the backward direction.
  217. *
  218. * @return true if characters remain, false otherwise.
  219. */
  220. public boolean backward()
  221. {
  222. return --position > lbound;
  223. }
  224. /**
  225. * Resets the sequence to its original position and discards all but the
  226. * initial match character.
  227. */
  228. public void reset()
  229. {
  230. position = start;
  231. matches.delete(1, length);
  232. }
  233. /**
  234. * Returns the lowercase character at the current iterator position.
  235. *
  236. * @return lowercase character at current position.
  237. */
  238. public char currentLower()
  239. {
  240. final int i;
  241. if (position < 0) {
  242. i = chars.length + position;
  243. } else if (position >= chars.length) {
  244. i = position - chars.length;
  245. } else {
  246. i = position;
  247. }
  248. return chars[i][0];
  249. }
  250. /**
  251. * Returns the uppercase character at the current iterator position.
  252. *
  253. * @return uppercase character at current position.
  254. */
  255. public char currentUpper()
  256. {
  257. final int i;
  258. if (position < 0) {
  259. i = chars.length + position;
  260. } else if (position >= chars.length) {
  261. i = position - chars.length;
  262. } else {
  263. i = position;
  264. }
  265. return chars[i][1];
  266. }
  267. /**
  268. * Adds the given character to the set of matched characters.
  269. *
  270. * @param c match character.
  271. */
  272. public void addMatchCharacter(final char c)
  273. {
  274. matches.append(c);
  275. }
  276. /**
  277. * Returns the number of matched characters.
  278. *
  279. * @return matched character count.
  280. */
  281. public int matchCount()
  282. {
  283. return matches.length();
  284. }
  285. /**
  286. * Returns the string of matched characters.
  287. *
  288. * @return match string.
  289. */
  290. public String matchString()
  291. {
  292. return matches.toString();
  293. }
  294. }
  295. }