ActionResults are a good way to make your controllers more testable. For example, HttpContext.Current is not available in unit tests a null reference exception occurs when testing code that calls HttpContext.Redirect, however because the Redirect ActionResult is returned and not executed by the controller unittests can check the action returned and ensure that it is a redirect without needing to mock an entire HttpContext.
Note: While MVC does wrap HttpContext which allows you to mock it while testing, it’s a lot easier if you don’t have to deal with it at all.
LoginAction
A similar approach can be used for LogIn/LogOut, you could abstract the FormsAuth call to a custom service but again you would need to pass a mock in your tests or risk another null reference, using a ActionResult instead makes testing much easier
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public ActionResult Login(string username, string password) { | |
if(username != “user” || password != “123”) { | |
ModelState.AddError(“username or password incorrect”); | |
return View(); | |
} | |
return new LoginResult(username); //I typically pass the actual user into the login result | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
[TestMethod] | |
public void Login_Success() { | |
AuthenticationController controller = new AuthenticationController(); | |
ActionResult result = controller.Login(“user”, “123”); | |
Assert.IsTrue(result.GetType() == typeof(LoginResult)); | |
Assert.AreEqual(“user”, ((LoginResult)result).Username); | |
} |
File Result
They also are a great method of code reuse. You would never imagine of putting the code that reads a view and renders it in every action, so why add the code that outputs a pdf/ical/rss/xls/graph/report inline in the controller? Simply return a Result that knows how to render the pdf and pass the data to it.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public ActionResult Graph() { | |
var data = null; //todo load some data | |
return new PdfResult(data); | |
} |
Source: https://github.com/MattHoneycutt/Fail-Tracker/blob/master/FailTracker.Web/ActionResults/StatusResult.cs
You can use the decorator pattern to extend what a view does. I like adding a status message at the top of a page after an action has taken place eg “User created successfully”. Instead of adding extra properties to every ViewModel to include a status message, I just put the message into TempData (Not ViewBag as that won’t live through a redirect) and consume it in the view.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
return new StatusResult(View(ViewModel), “user created succesfully”); |
I then use an extension method to make it a bit nicer to use
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
return View(ViewModel).WithSuccessMessage(“user created succesfully”); | |
return RedirectToAction(“Index”).WithSuccessMessage(“user created succesfully”); |
Unit tests can then check that a StatusResult is returned that contains another action which performs the redirect.