/Models/SaltedHash.cs

# · C# · 161 lines · 72 code · 14 blank · 75 comment · 0 complexity · c925d7d961f7ab1588b213e4135b16ec MD5 · raw file

  1. #region License Information
  2. // Copyright (c) 2005 - 2009, Allan Spartacus Mangune (allan@owlpal.com)
  3. // All rights reserved.
  4. //
  5. // Redistribution and use in source and binary forms, with or without modification,
  6. // are permitted provided that the following conditions are met:
  7. //
  8. // * Redistributions of source code must retain the above copyright notice, this list
  9. // of conditions and the following disclaimer.
  10. //
  11. // * Redistributions in binary form must reproduce the above copyright notice, this list of
  12. // conditions and the following disclaimer in the documentation and/or other materials provided
  13. // with the distribution.
  14. //
  15. // * Neither the name of Owlpal nor the names of its contributors may be used to endorse or
  16. // promote products derived from this software without specific prior written permission.
  17. //
  18. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS
  19. // OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
  20. // FITNESS FOR AllowHtmlInComment PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
  21. // BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  22. // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
  23. // OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  24. // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
  25. // USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. #endregion
  27. using System;
  28. using System.Security.Cryptography;
  29. namespace Owlpal.Security
  30. {
  31. public sealed class SaltedHash
  32. {
  33. #region Members
  34. private readonly string salt;
  35. private readonly string hash;
  36. private const int saltLength = 20;
  37. #endregion
  38. #region Properties
  39. /// <summary>
  40. ///
  41. /// </summary>
  42. public string Hash
  43. {
  44. get { return this.hash; }
  45. }
  46. /// <summary>
  47. ///
  48. /// </summary>
  49. public string Salt
  50. {
  51. get { return this.salt; }
  52. }
  53. #endregion
  54. #region Public Members
  55. /// <summary>
  56. ///
  57. /// </summary>
  58. /// <param name="password"></param>
  59. /// <returns></returns>
  60. public SaltedHash Create(string password)
  61. {
  62. string salt = CreateSalt();
  63. string hash = ComputeHash(salt, password);
  64. return new SaltedHash(salt, hash);
  65. }
  66. /// <summary>
  67. ///
  68. /// </summary>
  69. /// <param name="salt"></param>
  70. /// <param name="hash"></param>
  71. /// <returns></returns>
  72. public SaltedHash Create(string salt, string hash)
  73. {
  74. return new SaltedHash(salt, hash);
  75. }
  76. /// <summary>
  77. ///
  78. /// </summary>
  79. /// <param name="password"></param>
  80. /// <param name="salt"></param>
  81. /// <returns></returns>
  82. public bool IsValid(string password, string salt)
  83. {
  84. return this.hash.Equals(ComputeHash(salt, password));
  85. }
  86. /// <summary>
  87. ///
  88. /// </summary>
  89. /// <param name="password"></param>
  90. /// <returns></returns>
  91. public bool IsValid(string password)
  92. {
  93. return this.hash.Equals(ComputeHash(this.salt, password));
  94. }
  95. #endregion
  96. #region Private Members
  97. /// <summary>
  98. ///
  99. /// </summary>
  100. /// <returns></returns>
  101. private string CreateSalt()
  102. {
  103. return Convert.ToBase64String(CreateRandomBytes(saltLength));
  104. }
  105. /// <summary>
  106. ///
  107. /// </summary>
  108. /// <param name="len"></param>
  109. /// <returns></returns>
  110. private byte[] CreateRandomBytes(int len)
  111. {
  112. byte[] randomBytes = new byte[len];
  113. new RNGCryptoServiceProvider().GetBytes(randomBytes);
  114. return randomBytes;
  115. }
  116. /// <summary>
  117. ///
  118. /// </summary>
  119. /// <param name="salt"></param>
  120. /// <param name="password"></param>
  121. /// <returns></returns>
  122. private string ComputeHash(string salt, string password)
  123. {
  124. return Convert.ToBase64String(new SHA256CryptoServiceProvider()
  125. .ComputeHash(System.Text.Encoding.UTF8.GetBytes(salt + password)));
  126. }
  127. #endregion
  128. #region Contructors
  129. /// <summary>
  130. ///
  131. /// </summary>
  132. public SaltedHash()
  133. {
  134. }
  135. /// <summary>
  136. ///
  137. /// </summary>
  138. /// <param name="salt"></param>
  139. /// <param name="hash"></param>
  140. public SaltedHash(string salt, string hash)
  141. {
  142. this.salt = salt;
  143. this.hash = hash;
  144. }
  145. #endregion
  146. }
  147. }