Search code examples
azureazure-management-apimicrosoft-commerce-serverratecard-api

Error when trying to fetch azure subscription rate card and consumption usage


I am trying to fetch ratecard and uage for my pay as you go subscription. However consistently getting exception. Tried multipple options but none are working. Option 1 used to work flawlessly in the past but since last 4 weeks getting below exception

================================ Exception details =================================

System.AggregateException
HResult=0x80131500
Message=One or more errors occurred.
Source=mscorlib
StackTrace:at System.Threading.Tasks.Task.ThrowIfExceptional(Boolean includeTaskCanceledExceptions)
at System.Threading.Tasks.Task`1.GetResultCore(Boolean waitCompletionNotification)
at System.Threading.Tasks.Task1.get_Result()
Inner Exception 1:
TaskCanceledException: A task was canceled.

============== I tried multiple options but none worked. ==================

Option 1 Authenticate using Microsoft.IdentityModel.Clients.ActiveDirectory version 3.13.9 library and Microsoft.Commerce api ====================

    var url = "https://login.microsoftonline.com" ;
    var tenant = "mytenant.onmicrosoft.com" ;
    var resource = "https://management.core.windows.net" ;
     
    var clientSecret = "xxxx";
    var clientId = "client id of the app registerd in azure ad";
        
    var authenticationContext = 
        new AuthenticationContext(CombineUrl(url, tenant));
    ClientCredential clientCred = new ClientCredential(clientId, clientSecret);
    var authTask = 
        authenticationContext.AcquireTokenAsync(resource, clientCred);
    authTask.Wait();
    var token =  authTask.Result.Token;
            
    string  offerDurableId= "MS-AZR-0003p"; // also tried with "MS-AZR-0063p"
    string currency = "USD";
    string locale = "en-US";
    string  regionInfo =  "US"; 
    string SubscriptionId = "my subscription id"
    string APIVERSION =   "2015-06-01-preview"; 
               
    string url =  string.Format(
        "https://management.azure.com/subscriptions/{0}/providers" + `
        "/Microsoft.Commerce/RateCard?api-version={1}" + `
        "&$filter=OfferDurableId eq '{2}' and Currency eq '{3}'" + `
        "and Locale eq '{4}' and RegionInfo eq '{5}'", SubscriptionId, 
         APIVERSION, offerDurableId, currency, locale, regionInfo); `
    using (var request = new HttpRequestMessage(HttpMethod.Get, url))
    {
        var httpClient = new HttpClient() `
        request.Headers.Authorization = 
        new AuthenticationHeaderValue("Bearer", token); 
        HttpResponseMessage response = httpClient.SendAsync(request).Result; // 
        throws "task Canceled" exception
        var readTask = response.Content.ReadAsStringAsync(); `
        readTask.Wait(); `
        data= readTask.Result; `

   }

============ Option 2 using OAuth library Microsoft.Identity.Client version 4.31.0and Microsoft.Consumption API's


    var scopes = new List<string>();
    var TenantId = "tenant id as shown in azure portal for my registerd app"
    var loginAuthority = "https://login.microsoftonline.com";
    scopes.Add("https://management.azure.com/.default"); 
          
    var authResult = AuthClient.AcquireTokenForClient(scope.ToArray())
                                        .WithAuthority(loginAuthority, TenantId)
                                        .ExecuteAsync().GetAwaiter().GetResult();
        
    string token = authResult.AccessToken;
                    
    using (var request = new HttpRequestMessage(HttpMethod.Get, url))
    {
        request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
        HttpResponseMessage response = HTTPClient.SendAsync(request).Result; // returns blank
    }

```


===============================

I have been using Option 1 in the past and it always worked, not sure what broke and need help in fixing it.









Solution

  • Not sure on the error, I assumed that may be its a timeout issues as the data returned is in MB's. Tried increasing the timeout value to 3 mins instead of default and it worked. ' httpClient.TimeOut = new TimeSpan(0,3,0);`

    Tested multiple times and its working for now.