124

This is my first time ever using JSON as well as System.Net and the WebRequest in any of my applications. My application is supposed to send a JSON payload, similar to the one below to an authentication server:

{
  "agent": {                             
    "name": "Agent Name",                
    "version": 1                                                          
  },
  "username": "Username",                                   
  "password": "User Password",
  "token": "xxxxxx"
}

To create this payload, I used the JSON.NET library. How would I send this data to the authentication server and receive its JSON response back? Here is what I have seen in some examples, but no JSON content:

var http = (HttpWebRequest)WebRequest.Create(new Uri(baseUrl));
http.Accept = "application/json";
http.ContentType = "application/json";
http.Method = "POST";

string parsedContent = "Parsed JSON Content needs to go here";
ASCIIEncoding encoding = new ASCIIEncoding();
Byte[] bytes = encoding.GetBytes(parsedContent);

Stream newStream = http.GetRequestStream();
newStream.Write(bytes, 0, bytes.Length);
newStream.Close();

var response = http.GetResponse();

var stream = response.GetResponseStream();
var sr = new StreamReader(stream);
var content = sr.ReadToEnd();

However, this seems to be a lot of code compaired to using other languages I have used in the past. Am I doing this correctly? And how would I get the JSON response back so I can parse it?

Thanks, Elite.

Updated Code

// Send the POST Request to the Authentication Server
// Error Here
string json = await Task.Run(() => JsonConvert.SerializeObject(createLoginPayload(usernameTextBox.Text, password)));
var httpContent = new StringContent(json, Encoding.UTF8, "application/json");
using (var httpClient = new HttpClient())
{
    // Error here
    var httpResponse = await httpClient.PostAsync("URL HERE", httpContent);
    if (httpResponse.Content != null)
    {
        // Error Here
        var responseContent = await httpResponse.Content.ReadAsStringAsync();
    }
}
Trevor Reid
  • 3,310
  • 4
  • 27
  • 46
Hunter Mitchell
  • 7,063
  • 18
  • 69
  • 116
  • 2
    You can try `WebClient.UploadString(JsonConvert.SerializeObjectobj(yourobj))` or `HttpClient.PostAsJsonAsync` – L.B May 10 '14 at 20:58

5 Answers5

184

I found myself using the HttpClient library to query RESTful APIs as the code is very straightforward and fully async'ed. To send this JSON payload:

{
  "agent": {                             
    "name": "Agent Name",                
    "version": 1                                                          
  },
  "username": "Username",                                   
  "password": "User Password",
  "token": "xxxxxx"
}

With two classes representing the JSON structure you posted that may look like this:

public class Credentials
{
    public Agent Agent { get; set; }
    
    public string Username { get; set; }
    
    public string Password { get; set; }
    
    public string Token { get; set; }
}

public class Agent
{
    public string Name { get; set; }
    
    public int Version { get; set; }
}

You could have a method like this, which would do your POST request:

var payload = new Credentials { 
    Agent = new Agent { 
        Name = "Agent Name",
        Version = 1 
    },
    Username = "Username",
    Password = "User Password",
    Token = "xxxxx"
};

// Serialize our concrete class into a JSON String
var stringPayload = JsonConvert.SerializeObject(payload);

// Wrap our JSON inside a StringContent which then can be used by the HttpClient class
var httpContent = new StringContent(stringPayload, Encoding.UTF8, "application/json");

var httpClient = new HttpClient()
    
// Do the actual request and await the response
var httpResponse = await httpClient.PostAsync("http://localhost/api/path", httpContent);

// If the response contains content we want to read it!
if (httpResponse.Content != null) {
    var responseContent = await httpResponse.Content.ReadAsStringAsync();
    
    // From here on you could deserialize the ResponseContent back again to a concrete C# type using Json.Net
}
CodeCaster
  • 147,647
  • 23
  • 218
  • 272
Kai Eichinger
  • 2,137
  • 1
  • 15
  • 13
  • 5
    perfect, but what is the await Task.run(()? – Hunter Mitchell May 10 '14 at 21:27
  • 1
    `Task.Run` is a shortcut since `.NET 4.5` for `Task.Factory.StartNew`. In this case I async'ify the serialization. You can obviously also do that synchronously. I just figured that, depending on the payload, it may take a bit longer, so I wouldn't want to waste unnecessary resources there. – Kai Eichinger May 10 '14 at 21:32
  • You mean the HttpClient? Then yes, that's a library made by Microsoft! – Kai Eichinger May 10 '14 at 21:38
  • I keep getting error on the lines that contain `await`. It says something about async methods. – Hunter Mitchell May 10 '14 at 21:45
  • 1
    Yes, you need to mark your method with the `async` modifier and the return type needs to be either `Task` (the equivalent for `void`) or `Task` (the equivalent for `T`). e.g. `public async Task DoRequest() { /* ... */ }` or `public async Task DoRequest() { /* ... */ return "hello world"; }`. What .NET version are you targeting with your project? – Kai Eichinger May 10 '14 at 21:49
  • Let me post the current code i have right now in my question so you can get a better look. It is under Updated Code. I also added a comment `//Error here` where the errors where. – Hunter Mitchell May 10 '14 at 21:50
  • let us [continue this discussion in chat](http://chat.stackoverflow.com/rooms/52445/discussion-between-kai-eichinger-and-elite-gamer) – Kai Eichinger May 10 '14 at 21:57
  • In my codebase we inherit `StringContent` into a class `JsonContent`; the constructors take 1) string content, then call `base(content, Encoding.Utf8, "application/json")` and 2) string content, Encoding encoding, then make the appropriate `base()` call. – Dagrooms Apr 11 '17 at 20:19
  • 29
    You shouldn't use Task.Run on synchronous CPU bound methods as you're just firing off a new thread for no benefit! – Stephen Foster Nov 30 '17 at 14:24
  • 3
    You don't have to type out the `JsonProperty` for every property. Just use Json.Net's built in [CamelCasePropertyNamesContractResolver](https://www.newtonsoft.com/json/help/html/ContractResolver.htm) or a [custom `NamingStrategy`](https://stackoverflow.com/a/47839347/3312114) to customize the serialization process – Seafish Jun 17 '18 at 02:34
  • 9
    Side note: don't use a `using` with `HttpClient`. See: https://aspnetmonsters.com/2016/08/2016-08-27-httpclientwrong/ – maxshuty Aug 07 '18 at 12:49
  • 5
    With System.Net.Http.Formatting you have extension methods defined: "await httpClient.PostAsJsonAsync("api/v1/domain", csObjRequest)" – hB0 Oct 31 '18 at 04:01
  • Why don't you use `await JsonConvert.SerializeObjectAsync(payload)`? – stomy Sep 20 '19 at 00:31
  • If you want to skip the object serialization and use a string json you copied from elsewhere, you can escape the json into a string and then use it in the example above. E.g. `string stringPayload = "{\"test\": 123}";` – Cale Sweeney Apr 20 '22 at 19:27
30

Using the JSON.NET NuGet package and anonymous types, you can simplify what the other posters are suggesting:

// ...

string payload = JsonConvert.SerializeObject(new
{
    agent = new
    {
        name    = "Agent Name",
        version = 1,
    },

    username = "username",
    password = "password",
    token    = "xxxxx",
});

var client = new HttpClient();
var content = new StringContent(payload, Encoding.UTF8, "application/json");

HttpResponseMessage response = await client.PostAsync(uri, content);

// ...
Maximilian Burszley
  • 18,243
  • 4
  • 34
  • 63
9

You can build your HttpContent using the combination of JObject to avoid and JProperty and then call ToString() on it when building the StringContent:

        /*{
          "agent": {                             
            "name": "Agent Name",                
            "version": 1                                                          
          },
          "username": "Username",                                   
          "password": "User Password",
          "token": "xxxxxx"
        }*/

        JObject payLoad = new JObject(
            new JProperty("agent", 
                new JObject(
                    new JProperty("name", "Agent Name"),
                    new JProperty("version", 1)
                    ),
                new JProperty("username", "Username"),
                new JProperty("password", "User Password"),
                new JProperty("token", "xxxxxx")    
                )
            );

        using (HttpClient client = new HttpClient())
        {
            var httpContent = new StringContent(payLoad.ToString(), Encoding.UTF8, "application/json");

            using (HttpResponseMessage response = await client.PostAsync(requestUri, httpContent))
            {
                response.EnsureSuccessStatusCode();
                string responseBody = await response.Content.ReadAsStringAsync();
                return JObject.Parse(responseBody);
            }
        }
Jan Dolejsi
  • 1,389
  • 13
  • 25
  • How do you avoid `Exception while executing function. Newtonsoft.Json: Can not add Newtonsoft.Json.Linq.JProperty to Newtonsoft.Json.Linq.JArray` errors? – Jari Turkia Sep 20 '18 at 06:30
  • 1
    An HttpClient instance is not supposed to create with using construct. The instance should be created once and used throughout the application. This is because it uses its own connection pool. Your code tend mostly to throw SocketException. https://learn.microsoft.com/en-us/dotnet/api/system.net.http.httpclient?view=netframework-4.7.2 – Harun Diluka Heshan Nov 13 '18 at 12:32
9

You can also use the PostAsJsonAsync() method available in HttpClient()

var requestObj= JsonConvert.SerializeObject(obj);
HttpResponseMessage response = await client.PostAsJsonAsync($"endpoint",requestObj);
Igor
  • 349
  • 1
  • 6
  • 1
    Can you please add explanation what your code does and how it solves the issue? – Nilambar Sharma Jun 21 '19 at 06:02
  • 1
    You can take whatever the object you want to post and serialize it using the SerializeObject(); `var obj= new Credentials { Agent = new Agent { Name = "Agent Name", Version = 1 }, Username = "Username", Password = "User Password", Token = "xxxxx" };` And then without having to convert it to httpContent, you can use the PostAsJsonAsync() passing the endpoint URL and the converted JSON object itself. – Rukshala Weerasinghe Jun 22 '19 at 07:13
  • 3
    FYI there's no `PostAsJsonAsync()` on the `HttpClient` since .NET4.5 or so – PandaWood Jul 14 '21 at 00:22
0

System.Net.Http.Json.JsonContent exists to make serializing into json very easy (that is, if you're not already using or unable to use convenience methods like httpClient.PostAsync());

I was using IDownstreamApi with a POST method and needed to pass HttpContent so the code simply looks like:

        var jsonContent = JsonContent.Create(request);
        using var response = await _downstreamApi.CallApiForAppAsync("GraphApi", options =>
        {
            options.RelativePath = "directoryObjects/getByIds";
            options.HttpMethod = HttpMethod.Post;
        }, jsonContent, cancellationToken: cancellationToken);
Ben Sampica
  • 2,912
  • 1
  • 20
  • 25