Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

What's the "right way" to use HttpClient synchronously?

I used quote marks around "right way" because I'm already well aware that the right way to use an asynchronous API is to simply let the asynchronous behavior propagate throughout the entire call chain. That's not an option here.

I'm dealing with a very large and complicated system designed specifically to do batch processing synchronously in a loop.

The reason why suddenly I'm using HttpClient is because prior to now all data for the batch processing was gathered from a SQL database, and now we're adding a Web API call to the mix.

Yes, we're calling a Web API in a synchronously executing loop. I know. Rewriting the whole thing to be async just isn't an option. This is actually what we want to do. (We're minimizing the number of API calls as much as possible)

I actually did try to propagate the async behavior up the call chain, but then I found myself 50 files deep in changes, still with hundreds of compiler errors to resolve, and lost all hope. I am defeated.

So then, back to the question, given Microsoft's recommendation to never use WebRequest for new development and to instead use HttpClient, which offers only an asynchronous API, what am I to do?

Here is some pseudo-code of what I'm doing...

foreach (var thingToProcess in thingsToProcess) {     thingToProcess.ProcessStuff(); // This makes an API call } 

How do I implement ProcessStuff()?

My first implementation looked like this

public void ProcessStuff() {     var apiResponse = myHttpClient // this is an instance of HttpClient         .GetAsync(someUrl)         .Result;      // do some stuff with the apiResponse } 

I was told however, that calling .Result in this manner can result in deadlocks when it's called from something like ASP.NET due to the synchronization context.

Guess what, this batch process will be kicked off from an ASP.NET controller. Yes, again, I know, this is silly. When it runs from ASP.NET it's only "batch processing" one item instead of the whole batch, but I digress, it still gets called from ASP.NET and thus I'm concerned about deadlocks.

So what's the "right way" to handle this?

like image 446
spoonraker Avatar asked Nov 28 '18 22:11

spoonraker


People also ask

How do you call HttpClient synchronously?

Try the following: var task = Task. Run(() => myHttpClient. GetAsync(someUrl)); task.

What is the use of HttpClient?

An HttpClient can be used to send requests and retrieve their responses. An HttpClient is created through a builder . The builder can be used to configure per-client state, like: the preferred protocol version ( HTTP/1.1 or HTTP/2 ), whether to follow redirects, a proxy, an authenticator, etc.


1 Answers

Try the following:

var task = Task.Run(() => myHttpClient.GetAsync(someUrl));  task.Wait(); var response = task.Result; 

Use it only when you cannot use an async method.

This method is completely deadlock free as mentioned on the MSDN blog: ASP.Net–Do not use Task .Result in main context.

like image 194
soccer7 Avatar answered Nov 15 '22 22:11

soccer7