WORKS! In my brand-new job at Electronic Arts we are fortunate enough to work with Microsoft's latest technologies. We are developing web applications and, as every good developer out there I like to test as many things as it is possible and practical. In the last sprint I found myself with the problem of testing the routing configuration of a Web API REST service and also the routing configuration of some MVC controllers. A DuckDuckGo search returned a great [article](https://www.strathweb.com/2012/08/testing-routes-in-asp-net-web-api/) at [StrathWeb](https://www.strathweb.com) that solved the Web API side. Searching then for the MVC side, I found that, unsurprisingly, the famous library [MvcContrib](https://mvccontrib.codeplex.com/) has already a solution, but using a very different, fluent approach. I already had my Web API tests written with Philip's class (Philip is the author of StrathWeb), so I thought I would create a similar class that solved the problem, borrowing the necessary code from MvcContrib test helpers. The two classes were VERY similar, so I ended refactoring the common code in an abstract base class with an interface. Following is the code of the interface, the three classes and two sample tests that show how to build and use each one. ```csharp /// /// Route tester can test a route defined by URL, HTTP method and a lambda expression of a call to an action of a controller. /// public interface IRouteTester { /// /// Tests that an URL with an HTTP method is executed by an action of a controller. /// /// Controller type /// Return type of the action /// The URL to be tested /// The HTTP method /// The lambda expression that defines an action of a controller void TestRoute ( string url, HttpMethod method, Expression> expression); /// /// Tests that an URL with an HTTP method is executed by an action of a controller. /// /// Controller type /// The URL to be tested /// The HTTP method /// The lambda expression that defines an action of a controller void TestRoute ( string url, HttpMethod method, Expression> expression); } ``` ```csharp public abstract class BaseRouteTester : IRouteTester { /// /// Tests that an URL with an HTTP method is executed by an action of a controller. /// /// Controller type /// Return type of the action /// The URL to be tested /// The HTTP method /// The lambda expression that defines an action of a controller public void TestRoute (string url, HttpMethod method, Expression> expression) { this.TestRoute (url, method, GetMethodCall(expression)); } /// /// Tests that an URL with an HTTP method is executed by an action of a controller. /// /// Controller type /// The URL to be tested /// The HTTP method /// The lambda expression that defines an action of a controller public void TestRoute (string url, HttpMethod method, Expression> expression) { this.TestRoute (url, method, GetMethodCall(expression)); } /// /// Set ups route data and tests that the URL matches with some route. /// /// The URL /// The HTTP method protected abstract void SetupAndTestRouteData(string url, HttpMethod method); /// /// Get a route value /// /// /// protected abstract object GetValue(string key); /// /// Is a route value defined? /// /// /// protected abstract bool HasValue(string key); /// /// Gets the name of the controller corresponding to the matched route /// /// protected abstract string GetControllerName(); /// /// Gets the name of the action corresponding to the matched route /// /// protected abstract string GetActionName(); /// /// Gets the for a expression corresponding to an (void method) /// /// Type of the parameters /// The expression /// private static MethodCallExpression GetMethodCall (Expression> expression) { return expression.Body as MethodCallExpression; } /// /// Gets the for a expression corresponding to an (method returning a T) /// /// Type of the parameters /// Return type of the call /// The expression /// private static MethodCallExpression GetMethodCall (Expression> expression) { return expression.Body as MethodCallExpression; } /// /// Get the method name of a /// /// /// private string GetMethodName(MethodCallExpression method) { if (method != null) { return method.Method.Name; } throw new ArgumentException("Expression is wrong"); } /// /// Tests that an URL with an HTTP method is executed by an action of a controller. /// /// Controller type /// The URL to be tested /// The HTTP method /// Method call expression private void TestRoute (string url, HttpMethod method, MethodCallExpression methodCall) { // check route this.SetupAndTestRouteData(url, method); // check controller string expectedController = typeof(TController).Name; string actualController = this.GetControllerName(); if (expectedController != actualController) { throw new ControllerNotFoundException(expectedController, actualController); } // check action string actualAction = this.GetActionName(); string expectedAction = this.GetMethodName(methodCall); if (expectedAction != actualAction) { throw new ActionNotFoundException(expectedAction, actualAction); } // check parameters for (int i = 0; i < methodCall.Arguments.Count; i++) { ParameterInfo param = methodCall.Method.GetParameters()[i]; bool isReferenceType = !param.ParameterType.IsValueType; bool isNullable = isReferenceType || (param.ParameterType.UnderlyingSystemType.IsGenericType && param.ParameterType.UnderlyingSystemType.GetGenericTypeDefinition() == typeof(Nullable<>)); string controllerParameterName = param.Name; object actualValue = GetValue(controllerParameterName); object expectedValue = null; Expression expressionToEvaluate = methodCall.Arguments[i]; // If the parameter is nullable and the expression is a Convert UnaryExpression, // we actually want to test against the value of the expression's operand. if (expressionToEvaluate.NodeType == ExpressionType.Convert && expressionToEvaluate is UnaryExpression) { expressionToEvaluate = ((UnaryExpression)expressionToEvaluate).Operand; } switch (expressionToEvaluate.NodeType) { case ExpressionType.Constant: expectedValue = ((ConstantExpression)expressionToEvaluate).Value; break; case ExpressionType.New: case ExpressionType.MemberAccess: expectedValue = Expression.Lambda(expressionToEvaluate).Compile().DynamicInvoke(); break; } if (isNullable && (string)actualValue == string.Empty && expectedValue == null) { // The parameter is nullable so an expected value of '' is equivalent to null; continue; } // HACK: this is only sufficient while System.Web.Mvc.UrlParameter has only a single value. if (actualValue == UrlParameter.Optional || (actualValue != null && actualValue.ToString().Equals("System.Web.Mvc.UrlParameter"))) { actualValue = null; } if (expectedValue is DateTime) { actualValue = Convert.ToDateTime(actualValue); } else { expectedValue = expectedValue == null ? null : expectedValue.ToString(); } bool isOptional = methodCall.Method.GetParameters()[i].IsOptional; if ((actualValue == null) && isOptional) { return; } if (!Equals(actualValue, expectedValue)) { throw new ValueMismatchException(controllerParameterName, actualValue, expectedValue); } } } } ``` ```csharp /// /// Class that allows to test a Web API route. /// public class RestRouteTester : BaseRouteTester { private readonly HttpConfiguration config; private IHttpControllerSelector controllerSelector; private HttpControllerContext controllerContext; private IHttpRouteData routeData; private HttpRequestMessage request; public RestRouteTester(HttpConfiguration conf) { this.config = conf; } /// /// Gets the name of the action corresponding to the matched route /// /// protected override string GetActionName() { if (this.controllerContext.ControllerDescriptor == null) { this.GetControllerName(); } var actionSelector = new ApiControllerActionSelector(); var descriptor = actionSelector.SelectAction(this.controllerContext); return descriptor.ActionName; } /// /// Gets the name of the controller corresponding to the matched route /// /// protected override string GetControllerName() { var descriptor = this.controllerSelector.SelectController(this.request); this.controllerContext.ControllerDescriptor = descriptor; return descriptor.ControllerType.Name; } /// /// Set ups route data and tests that the URL matches with some route. /// /// The URL /// The HTTP method protected override void SetupAndTestRouteData(string url, HttpMethod method) { this.request = new HttpRequestMessage(method, url); this.routeData = config.Routes.GetRouteData(this.request); if (this.routeData == null) { throw new RouteNotFoundException(method, url); } this.request.Properties[HttpPropertyKeys.HttpRouteDataKey] = routeData; this.controllerSelector = new DefaultHttpControllerSelector(config); this.controllerContext = new HttpControllerContext(config, routeData, this.request); } /// /// Get a route value /// /// /// protected override object GetValue(string key) { foreach (var routeValueKey in this.routeData.Values.Keys) { if (string.Equals(routeValueKey, key, StringComparison.InvariantCultureIgnoreCase)) { if (this.routeData.Values[routeValueKey] == null) { return null; } return this.routeData.Values[routeValueKey].ToString(); } } return null; } /// /// Is a route value defined? /// /// /// protected override bool HasValue(string key) { return this.routeData.Values.ContainsKey(key); } } ``` ```csharp public class WebRouteTester : BaseRouteTester { private readonly RouteCollection routeCollection; private RouteData routeData; public WebRouteTester(RouteCollection routeCollection) { this.routeCollection = routeCollection; } /// /// Set ups route data and tests that the URL matches with some route. /// /// The URL /// The HTTP method protected override void SetupAndTestRouteData(string url, HttpMethod method) { this.routeData = routeCollection.GetRouteData(new FakeHttpContext(url, method.ToString())); if (this.routeData == null) { throw new RouteNotFoundException(method, url); } } /// /// Get a route value /// /// /// protected override object GetValue(string key) { foreach (var routeValueKey in this.routeData.Values.Keys) { if (string.Equals(routeValueKey, key, StringComparison.InvariantCultureIgnoreCase)) { if (this.routeData.Values[routeValueKey] == null) { return null; } if (this.routeData.Values[routeValueKey].GetType().Name == "UrlParameter") { return null; } return this.routeData.Values[routeValueKey]; } } return null; } /// /// Is a route value defined? /// /// /// protected override bool HasValue(string key) { return routeData.Values.ContainsKey(key); } /// /// Gets the name of the controller corresponding to the matched route /// /// protected override string GetControllerName() { return this.GetValue("controller") + "Controller"; } /// /// Gets the name of the action corresponding to the matched route /// /// protected override string GetActionName() { return this.GetValue("action").ToString(); } } ``` ```csharp [TestClass] public class WebRoutingTests { private RouteCollection routeCollection; [TestInitialize] public void Initialize() { this.routeCollection = new RouteCollection(); WebRouteConfig.RegisterRoutes(routeCollection); } [TestMethod] public void PlaylistController_GetTracksOffset() { TestRoute("~/playlist/45/GetTracks/56", HttpMethod.Get, (PlaylistController tc) => tc.GetTracks(45, 56, 100)); } #region Private methods private void TestRoute (string url, HttpMethod method, Expression> expression) { this.routeTester.TestRoute(url, method, expression); } private void TestRoute (string url, HttpMethod method, Expression> expression) { this.routeTester.TestRoute(url, method, expression); } #endregion } ```