153

I am trying to write some unit tests for my ApiController and faced some issues. There is a nice extension method called Request.CreateResponse that helps a lot with generating response.

public HttpResponseMessage Post(Product product)
{
  var createdProduct = repo.Add(product);
  return this.Request.CreateResponse(HttpStatusCode.Created, createdProduct);
}

Is there any way to mock CreateResponse without using of partial mocks or direct using of "new HttpResponseMessage(...)"?

3
  • 1
    Why do you want to mock the CreateResponse? Why not assert on the returned HttpResponseMessage Content and StatusCode properties that the correct values are set? Commented Jun 3, 2012 at 7:48
  • 3
    If I run this method form unit tests, it will fail with exception that configuration is not set. Commented Jun 3, 2012 at 8:01
  • 1
    Ok, shame on me, I just needed to write this.Request.CreateResponse(HttpStatusCode.Accepted, createdProduct, GlobalConfiguration.Configuration) Commented Jun 3, 2012 at 8:03

5 Answers 5

249

Another way to solve this is to do the following:

controller.Request = new HttpRequestMessage();
controller.Request.Properties.Add(HttpPropertyKeys.HttpConfigurationKey, 
                                  new HttpConfiguration());

If you are upgrading to webapi 5.0, then you'll need to change this to:

controller.Request = new HttpRequestMessage();
controller.Request.SetConfiguration(new HttpConfiguration());

The reason why you need to do this is because you have to have Request populated on the controller otherwise the extension methods on Request won't work. You also have to have an HttpConfiguration set on the Request otherwise routing and other parts of the pipeline won't function correctly.

Sign up to request clarification or add additional context in comments.

6 Comments

Yes, that might work as well, but it seems Request.CreateResponse also work.
Adding GlobalConfiguration.Configuration still left request null. Setting the Request worked for me, however this wasn't the end of my problems as my Action was also calling Url.Link and the route name was not defined.
Setting the ControllerContext after this call gives me that error too, but only with WebApi 5.0.0. But if I set the HttpConfiguration object on the Request last, it works fine.
This works perfectly, but I fail to understand why this works. Could you elaborate this in the answer?
|
24

You could set up the controller object for testability like this:

var config = new HttpConfiguration();
var request = new HttpRequestMessage(HttpMethod.Post, "http://localhost/api/products");
var route = config.Routes.MapHttpRoute("DefaultApi", "api/{controller}/{id}");
var routeData = new HttpRouteData(route, new HttpRouteValueDictionary { { "controller", "products" } });

controller.ControllerContext = new HttpControllerContext(config, routeData, request);
controller.Request = request;
controller.Request.Properties[HttpPropertyKeys.HttpConfigurationKey] = config;

Copied from Peter Provost's comprehensive blog post on Unit Testing ASP.NET Web API.

1 Comment

You may also be interested in the MS docs related to this: learn.microsoft.com/en-us/aspnet/web-api/overview/… You may also need to add references to System.Web.Http and System.Net.Http
8

For Web API 2, you can simply add

controller.Request = new HttpRequestMessage();
controller.Configuration = new HttpConfiguration();

Like so

[TestMethod]
public void GetReturnsProduct()
{
    // Arrange
    var controller = new ProductsController(repository);
    controller.Request = new HttpRequestMessage();
    controller.Configuration = new HttpConfiguration();

    // Act
    var response = controller.Get(10);

    // Assert
    Product product;
    Assert.IsTrue(response.TryGetContentValue<Product>(out product));
    Assert.AreEqual(10, product.Id);
}

It's important to set Request and Configuration on the controller. Otherwise, the test will fail with an ArgumentNullException or InvalidOperationException.

See here for more info.

Comments

1

WebAPI 1 here with a similar issue using VB.

I managed to hybrid responses here to get this to work as simple as this:

Dim request As HttpRequestMessage = New HttpRequestMessage()
Return request.CreateResponse(HttpStatusCode.BadRequest, myCustomClassObject, GlobalConfiguration.Configuration)

Just posting in case it helps anyone.

Comments

1

In your test class, create an instance of the controller class. e.g var customerController= new CustomerController();

customerController.Request = new HttpRequestMessage();
customerController.Request.Properties.Add(HttpPropertyKeys.HttpConfigurationKey, new HttpConfiguration());

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.