/trunk/SVN.Net/SVN.Net/Controllers/AccountController.cs

# · C# · 279 lines · 207 code · 53 blank · 19 comment · 29 complexity · c60c261a3c8a5b86aaf053c929f387e4 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Web;
  6. using System.Web.Mvc;
  7. using System.Web.Security;
  8. namespace SVN.Net.Controllers
  9. {
  10. [HandleError]
  11. public class AccountController : Controller
  12. {
  13. public AccountController()
  14. : this(null, null)
  15. {
  16. }
  17. public AccountController(IFormsAuthentication formsAuth, MembershipProvider provider)
  18. {
  19. FormsAuth = formsAuth ?? new FormsAuthenticationWrapper();
  20. Provider = provider ?? Membership.Provider;
  21. }
  22. public IFormsAuthentication FormsAuth
  23. {
  24. get;
  25. private set;
  26. }
  27. public MembershipProvider Provider
  28. {
  29. get;
  30. private set;
  31. }
  32. [Authorize]
  33. public ActionResult ChangePassword(string currentPassword, string newPassword, string confirmPassword)
  34. {
  35. ViewData["Title"] = "Change Password";
  36. ViewData["PasswordLength"] = Provider.MinRequiredPasswordLength;
  37. // Non-POST requests should just display the ChangePassword form
  38. if (Request.HttpMethod != "POST")
  39. {
  40. return View();
  41. }
  42. // Basic parameter validation
  43. List<string> errors = new List<string>();
  44. if (String.IsNullOrEmpty(currentPassword))
  45. {
  46. errors.Add("You must specify a current password.");
  47. }
  48. if (newPassword == null || newPassword.Length < Provider.MinRequiredPasswordLength)
  49. {
  50. errors.Add(String.Format(CultureInfo.InvariantCulture,
  51. "You must specify a new password of {0} or more characters.",
  52. Provider.MinRequiredPasswordLength));
  53. }
  54. if (!String.Equals(newPassword, confirmPassword, StringComparison.Ordinal))
  55. {
  56. errors.Add("The new password and confirmation password do not match.");
  57. }
  58. if (errors.Count == 0)
  59. {
  60. // Attempt to change password
  61. MembershipUser currentUser = Provider.GetUser(User.Identity.Name, true /* userIsOnline */);
  62. bool changeSuccessful = false;
  63. try
  64. {
  65. changeSuccessful = currentUser.ChangePassword(currentPassword, newPassword);
  66. }
  67. catch
  68. {
  69. // An exception is thrown if the new password does not meet the provider's requirements
  70. }
  71. if (changeSuccessful)
  72. {
  73. return RedirectToAction("ChangePasswordSuccess");
  74. }
  75. else
  76. {
  77. errors.Add("The current password is incorrect or the new password is invalid.");
  78. }
  79. }
  80. // If we got this far, something failed, redisplay form
  81. ViewData["errors"] = errors;
  82. return View();
  83. }
  84. public ActionResult ChangePasswordSuccess()
  85. {
  86. ViewData["Title"] = "Change Password";
  87. return View();
  88. }
  89. public ActionResult Login(string username, string password, bool? rememberMe)
  90. {
  91. ViewData["Title"] = "Login";
  92. // Non-POST requests should just display the Login form
  93. if (Request.HttpMethod != "POST")
  94. {
  95. return View();
  96. }
  97. // Basic parameter validation
  98. List<string> errors = new List<string>();
  99. if (String.IsNullOrEmpty(username))
  100. {
  101. errors.Add("You must specify a username.");
  102. }
  103. if (errors.Count == 0)
  104. {
  105. // Attempt to login
  106. bool loginSuccessful = Provider.ValidateUser(username, password);
  107. if (loginSuccessful)
  108. {
  109. FormsAuth.SetAuthCookie(username, rememberMe ?? false);
  110. return RedirectToAction("Index", "Home");
  111. }
  112. else
  113. {
  114. errors.Add("The username or password provided is incorrect.");
  115. }
  116. }
  117. // If we got this far, something failed, redisplay form
  118. ViewData["errors"] = errors;
  119. ViewData["username"] = username;
  120. return View();
  121. }
  122. public ActionResult Logout()
  123. {
  124. FormsAuth.SignOut();
  125. return RedirectToAction("Index", "Home");
  126. }
  127. public ActionResult Register(string username, string email, string password, string confirmPassword)
  128. {
  129. ViewData["Title"] = "Register";
  130. ViewData["PasswordLength"] = Provider.MinRequiredPasswordLength;
  131. // Non-POST requests should just display the Register form
  132. if (Request.HttpMethod != "POST")
  133. {
  134. return View();
  135. }
  136. // Basic parameter validation
  137. List<string> errors = new List<string>();
  138. if (String.IsNullOrEmpty(username))
  139. {
  140. errors.Add("You must specify a username.");
  141. }
  142. if (String.IsNullOrEmpty(email))
  143. {
  144. errors.Add("You must specify an email address.");
  145. }
  146. if (password == null || password.Length < Provider.MinRequiredPasswordLength)
  147. {
  148. errors.Add(String.Format(CultureInfo.InvariantCulture,
  149. "You must specify a password of {0} or more characters.",
  150. Provider.MinRequiredPasswordLength));
  151. }
  152. if (!String.Equals(password, confirmPassword, StringComparison.Ordinal))
  153. {
  154. errors.Add("The password and confirmation do not match.");
  155. }
  156. if (errors.Count == 0)
  157. {
  158. // Attempt to register the user
  159. MembershipCreateStatus createStatus;
  160. MembershipUser newUser = Provider.CreateUser(username, password, email, null, null, true, null, out createStatus);
  161. if (newUser != null)
  162. {
  163. FormsAuth.SetAuthCookie(username, false /* createPersistentCookie */);
  164. return RedirectToAction("Index", "Home");
  165. }
  166. else
  167. {
  168. errors.Add(ErrorCodeToString(createStatus));
  169. }
  170. }
  171. // If we got this far, something failed, redisplay form
  172. ViewData["errors"] = errors;
  173. ViewData["username"] = username;
  174. ViewData["email"] = email;
  175. return View();
  176. }
  177. public static string ErrorCodeToString(MembershipCreateStatus createStatus)
  178. {
  179. // See http://msdn.microsoft.com/en-us/library/system.web.security.membershipcreatestatus.aspx for
  180. // a full list of status codes.
  181. switch (createStatus)
  182. {
  183. case MembershipCreateStatus.DuplicateUserName:
  184. return "Username already exists. Please enter a different user name.";
  185. case MembershipCreateStatus.DuplicateEmail:
  186. return "A username for that e-mail address already exists. Please enter a different e-mail address.";
  187. case MembershipCreateStatus.InvalidPassword:
  188. return "The password provided is invalid. Please enter a valid password value.";
  189. case MembershipCreateStatus.InvalidEmail:
  190. return "The e-mail address provided is invalid. Please check the value and try again.";
  191. case MembershipCreateStatus.InvalidAnswer:
  192. return "The password retrieval answer provided is invalid. Please check the value and try again.";
  193. case MembershipCreateStatus.InvalidQuestion:
  194. return "The password retrieval question provided is invalid. Please check the value and try again.";
  195. case MembershipCreateStatus.InvalidUserName:
  196. return "The user name provided is invalid. Please check the value and try again.";
  197. case MembershipCreateStatus.ProviderError:
  198. return "The authentication provider returned an error. Please verify your entry and try again. If the problem persists, please contact your system administrator.";
  199. case MembershipCreateStatus.UserRejected:
  200. return "The user creation request has been canceled. Please verify your entry and try again. If the problem persists, please contact your system administrator.";
  201. default:
  202. return "An unknown error occurred. Please verify your entry and try again. If the problem persists, please contact your system administrator.";
  203. }
  204. }
  205. }
  206. // The FormsAuthentication type is sealed and contains static members, so it is difficult to
  207. // unit test code that calls its members. The interface and helper class below demonstrate
  208. // how to create an abstract wrapper around such a type in order to make the AccountController
  209. // code unit testable.
  210. public interface IFormsAuthentication
  211. {
  212. void SetAuthCookie(string userName, bool createPersistentCookie);
  213. void SignOut();
  214. }
  215. public class FormsAuthenticationWrapper : IFormsAuthentication
  216. {
  217. public void SetAuthCookie(string userName, bool createPersistentCookie)
  218. {
  219. FormsAuthentication.SetAuthCookie(userName, createPersistentCookie);
  220. }
  221. public void SignOut()
  222. {
  223. FormsAuthentication.SignOut();
  224. }
  225. }
  226. }