Using ADAL C# as Confidential User /Daemon Server /Server-to-Server - 401 Unauthorized

后端 未结 3 1764
眼角桃花
眼角桃花 2020-11-30 10:59

Refering to not answered Questions:

401- Unauthorized authentication using REST API Dynamics CRM with Azure AD

and

Dynamics CRM Online 2016 - Daemon

相关标签:
3条回答
  • 2020-11-30 11:10

    I finally found a solution. Provided by Joao R. in this Post:

    https://community.dynamics.com/crm/f/117/t/193506

    First of all: FORGET ADAL

    My problem was the whole time that I was using "wrong" URLS as it seems you need other adresses when not using Adal (or more general: user-redirect).


    Solution

    Construct following HTTP-Reqest for the Token:

    URL: https://login.windows.net/MyCompanyTenant.onmicrosoft.com/oauth2/token

    Header:

    • Cache-Control: no-cache
    • Content-Type: application/x-www-form-urlencoded

    Body:

    • client_id: YourClientIdFromAzureAd
    • resource: https://myCompanyTenant.crm.dynamics.com
    • username: yourServiceUser@myCompanyTenant.onmicrosoft.com
    • password: yourServiceUserPassword
    • grant_type: password
    • client_secret: YourClientSecretFromAzureAd

    Construct the following HTTP-Request for the access to WebApi:

    URL: https://MyCompanyTenant.api.crm.dynamics.com/api/data/v8.0/accounts

    Header:

    • Cache-Control: no-cache
    • Accept: application/json
    • OData-Version: 4.0
    • Authorization: Bearer TokenRetrievedFomRequestAbove

    Node.js Solution (Module for getting the Token)

    var https = require("https");
    var querystring = require("querystring");
    var config = require("../config/configuration.js");
    var q = require("q");
    
    var authHost = config.oauth.host;
    var authPath = config.oauth.path;
    var clientId = config.app.clientId;
    var resourceId = config.crm.resourceId;
    var username = config.crm.serviceUser.name;
    var password = config.crm.serviceUser.password;
    var clientSecret =config.app.clientSecret;
    
    function retrieveToken() {
        var deferred = q.defer();   
        var bodyDataString = querystring.stringify({
            grant_type: "password",
            client_id:  clientId, 
            resource: resourceId,
            username: username,
            password: password,        
            client_secret: clientSecret
        });
        var options = {
            host: authHost,
            path: authPath,
            method: 'POST',
            headers: {
                "Content-Type": "application/x-www-form-urlencoded",
                "Cache-Control": "no-cache"
            }
        };      
        var request = https.request(options, function(response){
            // Continuously update stream with data
            var body = '';
            response.on('data', function(d) {
                body += d;
            });
            response.on('end', function() {
                var parsed = JSON.parse(body); //todo: try/catch
                deferred.resolve(parsed.access_token);
            });               
        });
    
        request.on('error', function(e) {
            console.log(e.message);
            deferred.reject("authProvider.retrieveToken: Error retrieving the authToken: \r\n"+e.message);
        });
    
       request.end(bodyDataString);
       return deferred.promise;    
    }
    
    module.exports = {retrieveToken: retrieveToken};
    

    C#-Solution (Getting and using the Token)

      public class AuthenticationResponse
      {
        public string token_type { get; set; }
        public string scope { get; set; }
        public int expires_in { get; set; }
        public int expires_on { get; set; }
        public int not_before { get; set; }
        public string resource { get; set; }
        public string access_token { get; set; }
        public string refresh_token { get; set; }
        public string id_token { get; set; }
      }
    

    private static async Task<AuthenticationResponse> GetAuthenticationResponse()
    {
      List<KeyValuePair<string, string>> vals = new List<KeyValuePair<string, string>>();
    
      vals.Add(new KeyValuePair<string, string>("client_id", ClientId));
      vals.Add(new KeyValuePair<string, string>("resource", ResourceId));
      vals.Add(new KeyValuePair<string, string>("username", "yxcyxc@xyxc.onmicrosoft.com"));
      vals.Add(new KeyValuePair<string, string>("password", "yxcycx"));
      vals.Add(new KeyValuePair<string, string>("grant_type", "password"));
      vals.Add(new KeyValuePair<string, string>("client_secret", Password));
    
    
      string url = string.Format("https://login.windows.net/{0}/oauth2/token", Tenant);
    
      using (HttpClient httpClient = new HttpClient())
      {
        httpClient.DefaultRequestHeaders.Add("Cache-Control", "no-cache");
        HttpContent content = new FormUrlEncodedContent(vals);
        HttpResponseMessage hrm = httpClient.PostAsync(url, content).Result;
    
        AuthenticationResponse authenticationResponse = null;
        if (hrm.IsSuccessStatusCode)
        {
          Stream data = await hrm.Content.ReadAsStreamAsync();
          DataContractJsonSerializer serializer = new
        DataContractJsonSerializer(typeof(AuthenticationResponse));
          authenticationResponse = (AuthenticationResponse)serializer.ReadObject(data);
        }
        return authenticationResponse;
      }
    }
    
    private static async Task DataOperations(AuthenticationResponse authResult)
    {    
      using (HttpClient httpClient = new HttpClient())
      {
        httpClient.BaseAddress = new Uri(ResourceApiId);
        httpClient.Timeout = new TimeSpan(0, 2, 0); //2 minutes
        httpClient.DefaultRequestHeaders.Add("OData-MaxVersion", "4.0");
        httpClient.DefaultRequestHeaders.Add("OData-Version", "4.0");
        httpClient.DefaultRequestHeaders.Add("Cache-Control", "no-cache");
        httpClient.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
        httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", authResult.access_token);
    
        Account account = new Account();
        account.name = "Test Account";
        account.telephone1 = "555-555";
    
        string content = String.Empty;
        content = JsonConvert.SerializeObject(account, new JsonSerializerSettings() { DefaultValueHandling = DefaultValueHandling.Ignore });
        HttpRequestMessage request = new HttpRequestMessage(HttpMethod.Post, "api/data/v8.0/accounts");
        request.Content = new StringContent(content);
        request.Content.Headers.ContentType = MediaTypeHeaderValue.Parse("application/json");
        HttpResponseMessage response = await httpClient.SendAsync(request);
        if (response.IsSuccessStatusCode)
        {
          Console.WriteLine("Account '{0}' created.", account.name);
        }
        else
        {
          throw new Exception(String.Format("Failed to create account '{0}', reason is '{1}'."
            , account.name
            , response.ReasonPhrase));
        }
    (...)
    
    0 讨论(0)
  • 2020-11-30 11:24

    Thanks IntegerWolf for the detailed post/answer. I already wasted a lot of time trying to connect to the CRM Web API without any luck, until I ran into your post!

    Please be aware that ClientId in the code sample is the ClientId provided when registering your application in AAD. At first my connection failed, because in the explanation the value for client_id is YourTenantGuid, so I used my Office 365 TenantId, but this should be your AAD application ClientId.

    0 讨论(0)
  • 2020-11-30 11:26

    IntegerWolf's answer definitely pointed me in the right direction, but here's what ended up working for me:

    Discovering the Authorization Authority

    I ran the following code (in LINQPad) to determine the authorization endpoint to use for the Dynamics CRM instance to which I want my daemon/service/application to connect:

    AuthenticationParameters ap =
        AuthenticationParameters.CreateFromResourceUrlAsync(
                                    new Uri(resource + "/api/data/"))
                                .Result;
    
    return ap.Authority;
    

    resource is the URL of your CRM instance (or other app/service that's using ADAL), e.g. "https://myorg.crm.dynamics.com".

    In my case, the return value was "https://login.windows.net/my-crm-instance-tenant-id/oauth2/authorize". I suspect you can simply replace the tenant ID of your instance.

    Source:

    • Discover the authority at run time – Connect to Microsoft Dynamics 365 web services using OAuth

    Manually Authorizing the Daemon/Service/Application

    This was the crucial step for which I failed to find any help.

    I had to open the following URL in a web browser [formatted for easier viewing]:

    https://login.windows.net/my-crm-instance-tenant-id/oauth2/authorize?
       client_id=my-app-id
      &response_type=code
      &resource=https%3A//myorg.crm.dynamics.com
    

    When the page for that URL loaded, I logged in using the credentials for the user for which I wanted to run my daemon/service/app. I was then prompted to grant access to Dynamics CRM for the daemon/service/app as the user for which I logged-in. I granted access.

    Note that the login.windows.net site/app tried to open the 'home page' of my app that I setup in my app's Azure Active Directory registration. But my app doesn't actually have a home page so this 'failed'. But the above still seems to have successfully authorized my app's credentials to access Dynamics.

    Acquiring a Token

    Finally, the code below based on the code in IntegerWolf's answer worked for me.

    Note that the endpoint used is mostly the same as for the 'manual authorization' described in the previous section except that the final segment of the URL path is token instead of authorize.

    string AcquireAccessToken(
            string appId,
            string appSecretKey,
            string resource,
            string userName,
            string userPassword)
    {
        Dictionary<string, string> contentValues =
            new Dictionary<string, string>()
            {
                    { "client_id", appId },
                    { "resource", resource },
                    { "username", userName },
                    { "password", userPassword },
                    { "grant_type", "password" },
                    { "client_secret", appSecretKey }
            };
    
        HttpContent content = new FormUrlEncodedContent(contentValues);
    
        using (HttpClient httpClient = new HttpClient())
        {
            httpClient.DefaultRequestHeaders.Add("Cache-Control", "no-cache");
    
            HttpResponseMessage response =
                httpClient.PostAsync(
                            "https://login.windows.net/my-crm-instance-tenant-id/oauth2/token",
                            content)
                .Result
                //.Dump() // LINQPad output
                ;
    
            string responseContent =
                    response.Content.ReadAsStringAsync().Result
                    //.Dump() // LINQPad output
                    ;
    
            if (response.IsOk() && response.IsJson())
            {
                Dictionary<string, string> resultDictionary =
                    (new JavaScriptSerializer())
                    .Deserialize<Dictionary<string, string>>(responseContent)
                        //.Dump() // LINQPad output
                        ;
    
                return resultDictionary["access_token"];
            }
        }
    
        return null;
    }
    

    The code above makes use of some extension methods:

    public static class HttpResponseMessageExtensions
    {
        public static bool IsOk(this HttpResponseMessage response)
        {
            return response.StatusCode == System.Net.HttpStatusCode.OK;
        }
    
        public static bool IsHtml(this HttpResponseMessage response)
        {
            return response.FirstContentTypeTypes().Contains("text/html");
        }
    
        public static bool IsJson(this HttpResponseMessage response)
        {
            return response.FirstContentTypeTypes().Contains("application/json");
        }
    
        public static IEnumerable<string> FirstContentTypeTypes(
            this HttpResponseMessage response)
        {
            IEnumerable<string> contentTypes =
                 response.Content.Headers.Single(h => h.Key == "Content-Type").Value;
    
            return contentTypes.First().Split(new string[] { "; " }, StringSplitOptions.None);
        }
    }
    

    Using a Token

    To use a token with requests made with the HttpClient class, just add an authorization header containing the token:

    httpClient.DefaultRequestHeaders.Authorization =
        new AuthenticationHeaderValue("Bearer", accessToken);
    
    0 讨论(0)
提交回复
热议问题