/Projects/Appleseed.Framework/Content/Security/RandomPasswordLDAP.cs

https://code.google.com/p/appleseedapp/ · C# · 239 lines · 94 code · 37 blank · 108 comment · 16 complexity · c7a7a5ad727ff039d51d73839b740aed MD5 · raw file

  1. namespace Appleseed.Framework.Content.Security
  2. {
  3. using System;
  4. using System.Security.Cryptography;
  5. /// <summary>
  6. /// The random password ldap.
  7. /// </summary>
  8. public class RandomPasswordLdap
  9. {
  10. #region Constants and Fields
  11. /// <summary>
  12. /// The default max password length.
  13. /// </summary>
  14. private const int DefaultMaxPasswordLength = 10;
  15. /// <summary>
  16. /// The default min password length.
  17. /// </summary>
  18. private const int DefaultMinPasswordLength = 8;
  19. /// <summary>
  20. /// The password chars lcase.
  21. /// </summary>
  22. private const string PasswordCharsLcase = "abcdefgijkmnopqrstwxyz";
  23. /// <summary>
  24. /// The password chars numeric.
  25. /// </summary>
  26. private const string PasswordCharsNumeric = "23456789";
  27. /// <summary>
  28. /// The password chars special.
  29. /// </summary>
  30. private const string PasswordCharsSpecial = "*$-+?_&=!%{}/";
  31. /// <summary>
  32. /// The password chars ucase.
  33. /// </summary>
  34. private const string PasswordCharsUcase = "ABCDEFGHJKLMNPQRSTWXYZ";
  35. #endregion
  36. #region Public Methods
  37. /// <summary>
  38. /// Generates a random password.
  39. /// </summary>
  40. /// <returns>
  41. /// Randomly generated password.
  42. /// </returns>
  43. /// <remarks>
  44. /// The length of the generated password will be determined at
  45. /// random. It will be no shorter than the minimum default and
  46. /// no longer than maximum default.
  47. /// </remarks>
  48. public static string Generate()
  49. {
  50. return Generate(DefaultMinPasswordLength, DefaultMaxPasswordLength);
  51. }
  52. /// <summary>
  53. /// Generates a random password of the exact length.
  54. /// </summary>
  55. /// <param name="length">
  56. /// Exact password length.
  57. /// </param>
  58. /// <returns>
  59. /// Randomly generated password.
  60. /// </returns>
  61. public static string Generate(int length)
  62. {
  63. return Generate(length, length);
  64. }
  65. /// <summary>
  66. /// Generates a random password.
  67. /// </summary>
  68. /// <param name="minLength">
  69. /// Minimum password length.
  70. /// </param>
  71. /// <param name="maxLength">
  72. /// Maximum password length.
  73. /// </param>
  74. /// <returns>
  75. /// Randomly generated password.
  76. /// </returns>
  77. /// <remarks>
  78. /// The length of the generated password will be determined at
  79. /// random and it will fall with the range determined by the
  80. /// function parameters.
  81. /// </remarks>
  82. public static string Generate(int minLength, int maxLength)
  83. {
  84. // Make sure that input parameters are valid.
  85. if (minLength <= 0 || maxLength <= 0 || minLength > maxLength)
  86. {
  87. return null;
  88. }
  89. // Create a local array containing supported password characters
  90. // grouped by types. You can remove character groups from this
  91. // array, but doing so will weaken the password strength.
  92. var charGroups = new[]
  93. {
  94. PasswordCharsLcase.ToCharArray(),
  95. PasswordCharsUcase.ToCharArray(),
  96. PasswordCharsNumeric.ToCharArray(),
  97. PasswordCharsSpecial.ToCharArray()
  98. };
  99. // Use this array to track the number of unused characters in each
  100. // character group.
  101. var charsLeftInGroup = new int[charGroups.Length];
  102. // Initially, all characters in each group are not used.
  103. for (var i = 0; i < charsLeftInGroup.Length; i++)
  104. {
  105. charsLeftInGroup[i] = charGroups[i].Length;
  106. }
  107. // Use this array to track (iterate through) unused character groups.
  108. var leftGroupsOrder = new int[charGroups.Length];
  109. // Initially, all character groups are not used.
  110. for (var i = 0; i < leftGroupsOrder.Length; i++)
  111. {
  112. leftGroupsOrder[i] = i;
  113. }
  114. // Because we cannot use the default randomizer, which is based on the
  115. // current time (it will produce the same "random" number within a
  116. // second), we will use a random number generator to seed the
  117. // randomizer.
  118. // Use a 4-byte array to fill it with random bytes and convert it then
  119. // to an integer value.
  120. var randomBytes = new byte[4];
  121. // Generate 4 random bytes.
  122. var rng = new RNGCryptoServiceProvider();
  123. rng.GetBytes(randomBytes);
  124. // Convert 4 bytes into a 32-bit integer value.
  125. var seed = (randomBytes[0] & 0x7f) << 24 | randomBytes[1] << 16 | randomBytes[2] << 8 | randomBytes[3];
  126. // Now, this is real randomization.
  127. var random = new Random(seed);
  128. // This array will hold password characters.
  129. // Allocate appropriate memory for the password.
  130. var password = minLength < maxLength ? new char[random.Next(minLength, maxLength + 1)] : new char[minLength];
  131. // Index of the last non-processed group.
  132. var lastLeftGroupsOrderIdx = leftGroupsOrder.Length - 1;
  133. // Generate password characters one at a time.
  134. for (var i = 0; i < password.Length; i++)
  135. {
  136. // Index which will be used to track not processed character groups.
  137. // If only one character group remained unprocessed, process it;
  138. // otherwise, pick a random character group from the unprocessed
  139. // group list. To allow a special character to appear in the
  140. // first position, increment the second parameter of the Next
  141. // function call by one, i.e. lastLeftGroupsOrderIdx + 1.
  142. var nextLeftGroupsOrderIdx = lastLeftGroupsOrderIdx == 0 ? 0 : random.Next(0, lastLeftGroupsOrderIdx);
  143. // Index of the next character group to be processed.
  144. // Get the actual index of the character group, from which we will
  145. // pick the next character.
  146. var nextGroupIdx = leftGroupsOrder[nextLeftGroupsOrderIdx];
  147. // Index of the last non-processed character in a group.
  148. // Get the index of the last unprocessed characters in this group.
  149. var lastCharIdx = charsLeftInGroup[nextGroupIdx] - 1;
  150. // Index of the next character to be added to password.
  151. // If only one unprocessed character is left, pick it; otherwise,
  152. // get a random character from the unused character list.
  153. var nextCharIdx = lastCharIdx == 0 ? 0 : random.Next(0, lastCharIdx + 1);
  154. // Add this character to the password.
  155. password[i] = charGroups[nextGroupIdx][nextCharIdx];
  156. // If we processed the last character in this group, start over.
  157. if (lastCharIdx == 0)
  158. {
  159. charsLeftInGroup[nextGroupIdx] = charGroups[nextGroupIdx].Length;
  160. }
  161. else
  162. {
  163. // There are more unprocessed characters left.
  164. // Swap processed character with the last unprocessed character
  165. // so that we don't pick it until we process all characters in
  166. // this group.
  167. if (lastCharIdx != nextCharIdx)
  168. {
  169. var temp = charGroups[nextGroupIdx][lastCharIdx];
  170. charGroups[nextGroupIdx][lastCharIdx] = charGroups[nextGroupIdx][nextCharIdx];
  171. charGroups[nextGroupIdx][nextCharIdx] = temp;
  172. }
  173. // Decrement the number of unprocessed characters in
  174. // this group.
  175. charsLeftInGroup[nextGroupIdx]--;
  176. }
  177. // If we processed the last group, start all over.
  178. if (lastLeftGroupsOrderIdx == 0)
  179. {
  180. lastLeftGroupsOrderIdx = leftGroupsOrder.Length - 1;
  181. }
  182. else
  183. {
  184. // There are more unprocessed groups left.
  185. // Swap processed group with the last unprocessed group
  186. // so that we don't pick it until we process all groups.
  187. if (lastLeftGroupsOrderIdx != nextLeftGroupsOrderIdx)
  188. {
  189. var temp = leftGroupsOrder[lastLeftGroupsOrderIdx];
  190. leftGroupsOrder[lastLeftGroupsOrderIdx] = leftGroupsOrder[nextLeftGroupsOrderIdx];
  191. leftGroupsOrder[nextLeftGroupsOrderIdx] = temp;
  192. }
  193. // Decrement the number of unprocessed groups.
  194. lastLeftGroupsOrderIdx--;
  195. }
  196. }
  197. // Convert password characters into a string and return the result.
  198. return new string(password);
  199. }
  200. #endregion
  201. }
  202. }