Return content with IHttpActionResult for non-OK response
Asked Answered
E

17

221

For returning from a Web API 2 controller, I can return content with the response if the response is OK (status 200) like this:

public IHttpActionResult Get()
{
    string myResult = ...
    return Ok(myResult);
}

If possible, I want to use the built-in result types here when possible

My question is, for another type of response (not 200), how can I return a message (string) with it? For example, I can do this:

public IHttpActionResult Get()
{
    return InternalServerError();
}

but not this:

public IHttpActionResult Get()
{
    return InternalServerError("Message describing the error here");
}

Ideally, I want this to be generalized so that I can send a message back with any of the implementations of IHttpActionResult.

Do I need to do this (and build my response message):

public IHttpActionResult Get()
{
    HttpResponseMessage responseMessage = ...;
    return ResponseMessage(responseMessage);
}

or is there a better way?

Epiphragm answered 18/2, 2015 at 16:29 Comment(5)
how about this: #10733144Isoelectronic
could you not use ApiController.InternalServerError msdn.microsoft.com/en-us/library/dn292630(v=vs.118).aspxLovelady
@Milen, Thank you. Something like that might work. The part I don't like is it requires creating a different IHttpActionResult implementation for each existing implementation I want to be able to use.Epiphragm
@Ric, no, the parameter is an Exception. I want to set a message as a string. Also, this doesn't address a more general case where the code might not necessarily be internal server error.Epiphragm
@mayabelle: Did you see Shamil Yakupov`s answer? It's much simpler and concise that the accepted answer.Freeze
E
45

I ended up going with the following solution:

public class HttpActionResult : IHttpActionResult
{
    private readonly string _message;
    private readonly HttpStatusCode _statusCode;

    public HttpActionResult(HttpStatusCode statusCode, string message)
    {
        _statusCode = statusCode;
        _message = message;
    }

    public Task<HttpResponseMessage> ExecuteAsync(CancellationToken cancellationToken)
    {
        HttpResponseMessage response = new HttpResponseMessage(_statusCode)
        {
            Content = new StringContent(_message)
        };
        return Task.FromResult(response);
    }
}

... which can be used like this:

public IHttpActionResult Get()
{
   return new HttpActionResult(HttpStatusCode.InternalServerError, "error message"); // can use any HTTP status code
}

I'm open to suggestions for improvement. :)

Epiphragm answered 18/2, 2015 at 16:59 Comment(1)
Shamil Yakupov's answer is the best answer but only from inside the ApiController class - it needs to be rewritten as something like "return new NegotiatedContentResult<T>(code, new T(...), controller)" to be used from outside the controller class. In that case a solution such as this one above may be more readable.Tiresome
C
473

You can use this:

return Content(HttpStatusCode.BadRequest, "Any object");
Choreodrama answered 10/10, 2015 at 12:5 Comment(5)
Short and simple solution. Having more codes means more bugs and time consuming maintenance.Stanger
When im trying this, the returned value of code (where code is a string) in return Content(HttpStatusCode.OK, code) is encapsulated in " which is unexpected, is there any reason for this? eg the value which is returned is "\"value\"" I'm using mvc5Lintwhite
If you need to do this from outside the ApiController class then you can use: return new NegotiatedContentResult<T>(code, new T(...), controller)Tiresome
can i return it from a class library? What do I need to reference?Harebrained
the simplest answer is usually the best and most applicable oneArmistead
F
65

You can use HttpRequestMessagesExtensions.CreateErrorResponse (System.Net.Http namespace), like so:

public IHttpActionResult Get()
{
   return ResponseMessage(Request.CreateErrorResponse(HttpStatusCode.InternalServerError, "Message describing the error here"));
}

It is preferable to create responses based on the request to take advantage of Web API's content negotiation.

Fusspot answered 18/2, 2015 at 16:38 Comment(5)
Request.CreateErrorResponse returns an HttpResponseMessage, not IHttpActionResult. What you describe is good practice for creating an HttpResponseMessage but doesn't address my question. Thanks anyway!Epiphragm
@Epiphragm you can create IHttpActionResult concrete and wrapped those code like this:Stillbirth
This worked for me but I used Request.CreateResponse so that the error shows as a string instead of under Message key.Soldierly
I am getting an error, the snippet is failing. It says 'request' is null. I am trying to use Request.CreateResponse @FusspotTalton
@SheenaAgrawal This code can only be executed in the context of an HTTP request. If ApiController.Request is null it means you are not in the right context, or something is broken in your WebAPI architecture.Fusspot
E
45

I ended up going with the following solution:

public class HttpActionResult : IHttpActionResult
{
    private readonly string _message;
    private readonly HttpStatusCode _statusCode;

    public HttpActionResult(HttpStatusCode statusCode, string message)
    {
        _statusCode = statusCode;
        _message = message;
    }

    public Task<HttpResponseMessage> ExecuteAsync(CancellationToken cancellationToken)
    {
        HttpResponseMessage response = new HttpResponseMessage(_statusCode)
        {
            Content = new StringContent(_message)
        };
        return Task.FromResult(response);
    }
}

... which can be used like this:

public IHttpActionResult Get()
{
   return new HttpActionResult(HttpStatusCode.InternalServerError, "error message"); // can use any HTTP status code
}

I'm open to suggestions for improvement. :)

Epiphragm answered 18/2, 2015 at 16:59 Comment(1)
Shamil Yakupov's answer is the best answer but only from inside the ApiController class - it needs to be rewritten as something like "return new NegotiatedContentResult<T>(code, new T(...), controller)" to be used from outside the controller class. In that case a solution such as this one above may be more readable.Tiresome
R
18

You can also do:

return InternalServerError(new Exception("SOME CUSTOM MESSAGE"));
Reminiscent answered 14/5, 2015 at 13:22 Comment(1)
Yes but its a pain to get that message text backHarlie
B
12

Simple:

return ResponseMessage(Request.CreateErrorResponse(HttpStatusCode.InternalServerError, "Your message"));

Remember to reference System.Net.Http and System.Net.

Bedim answered 10/8, 2018 at 14:18 Comment(0)
M
7

Anyone who is interested in returning anything with any statuscode with returning ResponseMessage:

//CreateResponse(HttpStatusCode, T value)
return ResponseMessage(Request.CreateResponse(HttpStatusCode.XX, object));
Moro answered 6/12, 2015 at 14:17 Comment(0)
T
7

In ASP.NET Web API 2, you can wrap any ResponseMessage in a ResponseMessageResult:

public IHttpActionResult Get()
{
   HttpResponseMessage responseMessage = ...
   return new ResponseMessageResult(responseMessage);
}

In some cases this may be the simplest way to get the desired result, although generally it might be preferable to use the various results in System.Web.Http.Results.

Trochelminth answered 9/1, 2017 at 19:28 Comment(0)
P
6

I would recommend reading this post. There are tons of ways to use existing HttpResponse as suggested, but if you want to take advantage of Web Api 2, then look at using some of the built-in IHttpActionResult options such as

return Ok() 

or

return NotFound()

Choose the right return type for Web Api Controllers

Pint answered 1/6, 2018 at 17:50 Comment(0)
E
4

A more detailed example with support of HTTP code not defined in C# HttpStatusCode.

public class MyController : ApiController
{
    public IHttpActionResult Get()
    {
        HttpStatusCode codeNotDefined = (HttpStatusCode)429;
        return Content(codeNotDefined, "message to be sent in response body");
    }
}

Content is a virtual method defined in abstract class ApiController, the base of the controller. See the declaration as below:

protected internal virtual NegotiatedContentResult<T> Content<T>(HttpStatusCode statusCode, T value);
Echopraxia answered 7/8, 2019 at 18:37 Comment(0)
D
2

Below code / class really helpful to handle all type of responses. May be success or fail etc.

Why should I use this? :

Suppose I am consuming web API service. There are some possibilities of output as below:

  1. You might not get any result because of validation error
  2. You will get expected result
  3. You will get error.

So here I have solution to handle all the scenarios. Also I tried to maintain uniformity in the output. You can give remark or actual error message. The web service consumer can only check IsSuccess true or not else will sure there is problem, and act as per situation. There should be perfect communication between web API developer and web API consumer. If result is not generating then why. The comment or exception will acknowledge and they will act accordingly. Hence this is best solution for web API consumer / user.

Single response for all all type of results.

  public class Response
    {
        /// <summary>
        /// Gets or sets a value indicating whether this instance is success.
        /// </summary>
        /// <value>
        /// <c>true</c> if this instance is success; otherwise, <c>false</c>.
        /// </value>
        public bool IsSuccess { get; set; } = false;
    
        /// <summary>
        /// Actual response if succeed 
        /// </summary>
        /// <value>
        /// Actual response if succeed 
        /// </value>
        public object Data { get; set; } = null;
    
        /// <summary>
        /// Remark if anythig to convey
        /// </summary>
        /// <value>
        /// Remark if anythig to convey
        /// </value>
        public string Remark { get; set; } = string.Empty;
        /// <summary>
        /// Gets or sets the error message.
        /// </summary>
        /// <value>
        /// The error message.
        /// </value>
        public object ErrorMessage { get; set; } = null;
    
       
    }  
    



[HttpGet]
        public IHttpActionResult Employees()
        {
            Response _res = new Response();
            try
            { 
                DalTest objDal = new DalTest(); 
                _res.Data = objDal.GetTestData();
                _res.IsSuccess = true;
                return Request.CreateResponse(HttpStatusCode.OK, _res);
            }
            catch (Exception ex)
            {
                _res.IsSuccess = false;
                _res.ErrorMessage = ex;
                return ResponseMessage(Request.CreateResponse(HttpStatusCode.InternalServerError, _res )); 
            } 
        }

You are welcome to give suggestion if any :)

I am happy if anyone is using and get benefited. Write me if any addition or modification makes this more improved, on [email protected].

Dumbhead answered 27/4, 2018 at 9:5 Comment(0)
S
1

@mayabelle you can create IHttpActionResult concrete and wrapped those code like this:

public class NotFoundPlainTextActionResult : IHttpActionResult
{
    public NotFoundPlainTextActionResult(HttpRequestMessage request, string message)
    {
        Request = request;
        Message = message;
    }

    public string Message { get; private set; }
    public HttpRequestMessage Request { get; private set; }

    public Task<HttpResponseMessage> ExecuteAsync(CancellationToken cancellationToken)
    {
        return Task.FromResult(ExecuteResult());
    }

    public HttpResponseMessage ExecuteResult()
    {
        var response = new HttpResponseMessage();

        if (!string.IsNullOrWhiteSpace(Message))
            //response.Content = new StringContent(Message);
            response = Request.CreateErrorResponse(HttpStatusCode.NotFound, new Exception(Message));

        response.RequestMessage = Request;
        return response;
    }
}
Stillbirth answered 3/3, 2015 at 9:20 Comment(0)
A
1

this answer is based on Shamil Yakupov answer, with real object instead of string.

using System.Dynamic;

dynamic response = new ExpandoObject();
response.message = "Email address already exist";

return Content<object>(HttpStatusCode.BadRequest, response);
Arrhenius answered 18/11, 2017 at 10:22 Comment(1)
Content<T> is very usefulArmistead
C
1

If you are just looking for sending empty response with specific HTTP code:

return StatusCode(StatusCodes.Status418ImATeapot);
Claudell answered 22/2, 2023 at 14:30 Comment(0)
B
0

I had the same problem. I want to create custom result for my api controllers, to call them like return Ok("some text");

Then i did this: 1) Create custom result type with singletone

public sealed class EmptyResult : IHttpActionResult
{
    public Task<HttpResponseMessage> ExecuteAsync(CancellationToken cancellationToken)
    {
        return Task.FromResult(new HttpResponseMessage(System.Net.HttpStatusCode.NoContent) { Content = new StringContent("Empty result") });
    }
}

2) Create custom controller with new method:

public class CustomApiController : ApiController
{
    public IHttpActionResult EmptyResult()
    {
        return new EmptyResult();
    }
}

And then i can call them in my controllers, like this:

public IHttpActionResult SomeMethod()
    {
       return EmptyResult();
    }
Bide answered 28/6, 2017 at 16:59 Comment(0)
H
0

For exceptions, I usually do

 catch (Exception ex)
        {
            return InternalServerError(new ApplicationException("Something went wrong in this request. internal exception: " + ex.Message));
        }
Heavy answered 14/2, 2018 at 2:32 Comment(0)
B
-1

Sorry for the late answer why don't you simple use

return BadRequest("your message");

I use it for all my IHttpActionResult errors its working well

here is the documentation : https://msdn.microsoft.com/en-us/library/system.web.http.apicontroller.badrequest(v=vs.118).aspx

Bushey answered 5/11, 2016 at 23:32 Comment(2)
Because not all errors are the result of bad requests, so a 400 response would be inappropriate. OP specifically gave a 500 response as an example.Fusspot
Yes it's only possible with BadRequest the other types don't take a message argumentBushey
D
-1

You can use this:

return BadRequest("No data is present in table!!");
Degroot answered 25/3, 2021 at 6:11 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.