Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to use HttpClient to send content in body of GET request?

Currently to send a parameterized GET request to an API interface I am writing the following code:

api/master/city/filter?cityid=1&citycode='ny' 

But I see that there is a limit on the URL length of 2,083 characters.

To avoid this I would like to send the parameters in json format in the content body for a GET request.

However, I see that none of the Get methods for the HttpClient allow for a content body to be sent. For the POST I could see there is a method within HttpClient named PostAsync that allows for a content body.

Is there a way to send parameters for a GET request not in the URL in order to avoid the URL length limit?

like image 498
John Avatar asked Apr 15 '17 01:04

John


People also ask

Can you send data in the body of a GET request?

Requests using GET should only be used to request data (they shouldn't include data). Note: Sending body/payload in a GET request may cause some existing implementations to reject the request — while not prohibited by the specification, the semantics are undefined.


2 Answers

Please read the caveats at the end of this answer as to why HTTP GET requests with bodies are, in general, not advised.


  • If you are using .NET Core, the standard HttpClient can do this out-of-the-box. For example, to send a GET request with a JSON body:

      HttpClient client = ...    ...    var request = new HttpRequestMessage   {       Method = HttpMethod.Get,       RequestUri = new Uri("some url"),       Content = new StringContent("some json", Encoding.UTF8, MediaTypeNames.Application.Json /* or "application/json" in older versions */),   };    var response = await client.SendAsync(request).ConfigureAwait(false);   response.EnsureSuccessStatusCode();    var responseBody = await response.Content.ReadAsStringAsync().ConfigureAwait(false); 
  • .NET Framework doesn't support this out-of-the-box (you will receive a ProtocolViolationException if you try the above code). Thankfully Microsoft has provided the System.Net.Http.WinHttpHandler package that does support the functionality - simply install and use it instead of the default HttpClientHandler when constructing your HttpClient instances:

      var handler = new WinHttpHandler();   var client = new HttpClient(handler);    <rest of code as above> 

    Reference: https://github.com/dotnet/runtime/issues/25485#issuecomment-467261945


Caveats:

  • HTTP GET with a body is a somewhat unconventional construct that falls in a gray area of the HTTP specification - the end result is that many older pieces of software either cannot handle such a request at all, or will explicitly reject it because they believe it to be malformed. You need to make very sure that the endpoint you're trying to send such a request to does support it, or at best you will get an HTTP error code back; at worst the body will be silently discarded. This can lead to some head-scratching debugging!
  • Caching proxy servers, again particularly older ones, may cache GET requests based only on the URL because they don't expect a body to be present. This could either result in the least recent request being cached forever (which will break your software), or that the only request ever cached is the most recent one issued (which will prevent caching from working as intended). Again, this can be very painful to figure out.
like image 113
Ian Kemp Avatar answered Oct 12 '22 12:10

Ian Kemp


I can't use .NET core and I don't want to install System.Net.Http.WinHttpHandler, which has a ton of dependencies. I solved it by using reflection, to trick WebRequest that it is legal to send body with a GET request (which is according to latest RFC). What I do is to set ContentBodyNotAllowed to false for HTTP verb "GET".

var request = WebRequest.Create(requestUri);  request.ContentType = "application/json"; request.Method = "GET";  var type = request.GetType(); var currentMethod = type.GetProperty("CurrentMethod", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(request);  var methodType = currentMethod.GetType(); methodType.GetField("ContentBodyNotAllowed", BindingFlags.NonPublic | BindingFlags.Instance).SetValue(currentMethod, false);  using (var streamWriter = new StreamWriter(request.GetRequestStream())) {     streamWriter.Write("<Json string here>"); }  var response = (HttpWebResponse)request.GetResponse(); 

Note, however, that the attribute ContentBodyNotAllowed belongs to a static field, so when its value changes, it remains in effect for the rest of the program. That's not a problem for my purposes.

like image 43
alfoks Avatar answered Oct 12 '22 12:10

alfoks