I have a service layer project on an MVC 5 ASP.NET application I am creating on .NET 4.5.2 which calls out to an External 3rd Party WCF Service to Get Information asynchronously. An original method to call external service was as below (there are 3 of these all similar in total which I call in order from my GetInfoFromExternalService method (note it isnt actually called that - just naming it for illustration)
private async Task<string> GetTokenIdForCarsAsync(Car[] cars)
{
try
{
if (_externalpServiceClient == null)
{
_externalpServiceClient = new ExternalServiceClient("WSHttpBinding_IExternalService");
}
string tokenId= await _externalpServiceClient .GetInfoForCarsAsync(cars).ConfigureAwait(false);
return tokenId;
}
catch (Exception ex)
{
//TODO plug in log 4 net
throw new Exception("Failed" + ex.Message);
}
finally
{
CloseExternalServiceClient(_externalpServiceClient);
_externalpServiceClient= null;
}
}
So that meant that when each async call had completed the finally block ran - the WCF client was closed and set to null and then newed up when another request was made. This was working fine until a change needed to be made whereby if the number of cars passed in by User exceeds 1000 I create a Split Function and then call my GetInfoFromExternalService method in a WhenAll with each 1000 - as below:
if (cars.Count > 1000)
{
const int packageSize = 1000;
var packages = SplitCarss(cars, packageSize);
//kick off the number of split packages we got above in Parallel and await until they all complete
await Task.WhenAll(packages.Select(GetInfoFromExternalService));
}
However this now falls over as if I have 3000 cars the method call to GetTokenId news up the WCF service but the finally blocks closes it so the second batch of 1000 that is attempting to be run throws an exception. If I remove the finally block the code works ok - but it is obviously not good practice to not be closing this WCF client.
I had tried putting it after my if else block where the cars.count is evaluated - but if a User uploads for e.g 2000 cars and that completes and runs in say 1 min - in the meantime as the user had control in the Webpage they could upload another 2000 or another User could upload and again it falls over with an Exception.
Is there a good way anyone can see to correctly close the External Service Client?
Based on the related question of yours, your "split" logic doesn't seem to give you what you're trying to achieve.
WhenAll
still executes requests in parallel, so you may end up running more than 1000 requests at any given moment of time. UseSemaphoreSlim
to throttle the number of simultaneously active requests and limit that number to 1000. This way, you don't need to do any splits.Another issue might be in how you handle the creation/disposal of
ExternalServiceClient
client. I suspect there might a race condition there.Lastly, when you re-throw from the
catch
block, you should at least include a reference to the original exception.Here's how to address these issues (untested, but should give you the idea):
Updated based on the comment:
The changes are minimal (but untested). First:
Then, using Marc Gravell's
ChunkExtension.Chunkify
, we introduceGetAllTokenIdForCarsAsync
, which in turn will be callingGetTokenIdForCarsAsync
from above:Now you can pass all 7000 cars into
GetAllTokenIdForCarsAsync
. This is a skeleton, it can be improved with some retry logic if any of the batch requests has failed (I'm leaving that up to you).