Skip to main content
Post a question

Notifications

Community site session details

Community site session details

Session Id : /UMtAqag3vZXqG+RP3E/1/
Dynamics 365 Community / Blogs / AX for Retail / Support for Service to Serv...

Support for Service to Service authentication in Retail Server

SergeyP Profile Picture SergeyP 2,928

Starting AX7 FallUpdate CTP2 Retail Server supports Service to Service authentication which makes it possible to have services capable of communicating with Retail Server without a need of a user in front of a screen to provide credentials at some point in time. This enables scenarios when you need to create processes to periodically contact Retail Server to perform different kind of tasks, for instance, synchronizing customers, orders, publishing products for eCommerce functionality and so on.

That feature leverages Azure AD with OAUTH 2 Client Credentials Grant, the details of this authentication flow can be found in Daemon app that calls a web API in the daemon's name. To setup the feature register 2 AAD applications in your tenant, one app will represent Retail Server's identity and another one the Client application making calls to the Retail Server. These are detailed steps:

Note: The values you will provide for the Issuer, Client ID and Server Resource ID will be compared against values in the Security Token by leveraging case sensitive strings comparisons, so, even if something is looking as a Url don't make an assumption that let's say https://contoso.com and https://contoso.com/ (note the trailing slash in the second one) are identical - they are not in fact.

1. Create AAD applications

a) Navigate to https://aad.portal.azure.com/

b) Click the menu item (on left hand side) Azure Active Directory and then App registrations -> New registration, you will see something like this:

4456.pastedimage1604907088094v1.png

Provide any descriptive name you want, put there something identifying the app belongs to the Retail Server, this will help you identifying it later. Keep all other settings unchanged with their default values as shown above.

c) Once you click “Register” button at the above form, click the link “Expose an API” and then “Add a scope”, copy the value which begins with “api://…” into the clipboard, you will need it later, that is “Application ID URI” or “Server Resource ID”. Click Save and Continue:

1351.pastedimage1604907714427v2.png

d) In the “Add a Scope” screen provide values like, for instance, below:

1602.pastedimage1604908060164v4.png

And then click the button “Add scope” on bottom. This completes registering AAD application corresponding to Retail Server.

e) Now initiate another App Registration, this time for the Client (if you don’t’ have one already). To do this proceed with the step (b), with only change by providing a different name. Once the app is created click the link “API permissions” and then click a button “Add a permission”. In the window opened select “APIs my organization uses” and then paste the name of the RS app created in previous step, in my case it is D365RetailServerSergTest:

1411.pastedimage1604908496394v6.png

f) Click on the application and you will see another windows opened, select the checkbox near the Permission name and click “Add Permission”:

0407.pastedimage1604908767761v7.png

g) Once you save the changes you should see something like this:

8037.pastedimage1604909002038v9.png

h) Now create the secret (or assign a certificate) for just setup Client application, for details please see Authentication: Two options

2. Add the applications' details into RS's authentication allowed list

a) In AX UI navigate to the form Commerce Shared Parameters, if you don't know where it is located you can type that name in the search box and once found click [ENTER]. Once on that form click Identity Providers tab, you will see a pane displaying 3 grids.

b) In the first grid (Identity Providers) select a row with type Azure Active Directory, this selection will result in the 2nd grid (Relying Parties) displaying list of client applications registered for the given identity provider. Click Add button in that 2nd grid and paste there the following values:

 - for the ClientId field specify the ID of the application  you created in the step #1, that is the GUID you can find in Azure portal for the Client (not Retail Server) AAD application you created above. Specific location is in the section Overview, the parameter Application (client) ID.

8371.pastedimage1604911421925v13.png

 - for the Type select Confidential

 - for the UserType specify Application

  To save this row's changes click on any other row in this grid and then click back the row you just added.

c) Scroll down to the Server Resource IDs grid, this one contains RS Uris allowed to be accessed by the application in Relying Parties grid. Click Add in the Server Resource IDs grid and add/fill out the cell Server Resource ID with the value corresponding to the Server Resource ID of your Retail Server (not Client) application which can be seen in the Azure Portal once you navigate to your Retail Server application's Overview section:

2816.pastedimage1604911529896v14.png

As a result you should have something like this:

pastedimage1604912339744v1.png

To save the changes either click on any other tab in the form or just close the form.

To bring the changes into the channel DB go to Retail and commerce->Retail IT->Distribution schedule and then execute the job 1110 (Global configuration). Wait until the job finishes its work, then, in addition, if you don't want to wait until cache expires, and if that is not production environment - you can execute iisreset

You have setup everything on AAD and AX sides, now it is time to test it in action. The LCS VM is shipped with Retail SDK which includes sample eCommerce Publishing client which leverages Service To Service authentication while contacting RS. So, open the project J:\RetailSDK\SampleExtensions\OnlineStore\Publishing\Storefront.Publishing.csproj in Visual Studio (Run Visual Studio as Admin and then load that project), once solution is loaded set the project Storefront.Publishing as Startup one, then open its app.config and provide values for the 4 parameters: 

aadClientId - specify the CLIENT ID corresponding to the Client AAD application you created earlier.

aadClientSecret - specify the client secret corresponding to the above Client AAD application. NOTE that password is accepted/stored here in clear text, this should never be done in PROD system.

aadAuthority - put here the value seen in the column Issuer of the grid Identity Providers, that value should look like https://sts.windows.net/<YourTenantGuidIsHere>/

Make sure to have the guid corresponding to your tenant instead of <YourTenantGuidIsHere>

retailServerUrl - substitute the value there with the one corresponding to the instance of your Retail Server, you can see that Url in LCS portal or just go to IIS, select the site RetailServer and then click Browse, you will see your browser opens with the url something like this: https://<ValueUniqueToYourDeploymentIsHere>.cloudax.dynamics.com/

Add there the suffix Commerce and as a result you have Retail Server url to be stored in the app.config

AppConfig.png

Now run the application and watch its console output - it will print out details while communicating with AAD and Retail Server.

While building your own client application it makes sense to leverage Retail Server Proxy which hides from you all the details related to transport and authentication. The method CreateManagerFactory located in the file J:\RetailSDK\OnlineStore\Ecommerce.Sdk.Core\Publishing\Publisher.cs shows how to get a token from AAD and then instantiates the proxy.

If you are customizing Retail Server and going to allow the Controller's method to accept calls with this authentication type you need to decorate the method with CommerceAuthorization attribute and role Application, you can also add as many roles as needed in your specific case, for instance the method below accepts calls from all supported by RS types: Employee, Anonymous, Customer, Application.

  1. [CommerceAuthorization(CommerceRoles.Employee, CommerceRoles.Anonymous, CommerceRoles.Customer, CommerceRoles.Application)]
  2. [HttpPost]
  3. public ChannelConfiguration GetOrgUnitConfiguration()
  4. {
  5. return this.channelManager.GetChannelConfiguration();
  6. }

Comments

*This post is locked for comments

  • SergeyP Profile Picture SergeyP 2,928
    Posted 22 Sep 2017 at 04:55:16

    Hi VKJ, Demo LCS VM, the one which hosts AX, contains Web Site called RetailStoreFront, that is sample online store which demonstrates how to consume Retail Server in C2 (customer) context, in fact that is basic end to end online store which allows to browse categories, products, create customers, orders, including delivery options and payments plus order history and a Wish List. The sources are available at J:\RetailSDK\OnlineStore. That entire solution is about using Retail Server (which is a host for CRT) by an external application, probably that is what you were looking for. That web site leverages Anonymous and C2 contexts explained in community.dynamics.com/.../basics-of-building-native-client-capable-of-c2-authentication-with-retail-server

  • Community Member Profile Picture Community Member
    Posted 21 Sep 2017 at 03:47:02

    Hi Sergey,

    Many thanks for your reply. One last question. I am working on a connector to a ecommerce(magento) application and I wanted to use CRT services to pull customers/products/orders etc.  and I was wondering if extending Retail Server is the only option or If I can consume CRT services directly.if "Yes" Is there any example available on how to consume CRT service in my own project. I have seen the SDK samples and though they talk about extending the CRT but no where they mention about how to use CRT by external application. My application is of type visual studio class library.

    Best Regards,

    Vivek

  • SergeyP Profile Picture SergeyP 2,928
    Posted 20 Sep 2017 at 06:38:20

    Hi VKJ, it is expected that ICustomerManager.Create fails with Unauthorized for Application context - that operation is not allowed for Application context. In current implementation it is only allowed for C1 (means employee who is using POS) and eCommerce cases (explanations and an example can be found in community.dynamics.com/.../basics-of-building-native-client-capable-of-c2-authentication-with-retail-server). Current implementation of AX7 doesn't expose any customer related APIs to application context, if you need them you need to create an extension for Retail Server.

  • Community Member Profile Picture Community Member
    Posted 15 Sep 2017 at 19:09:24

    Hi Sergey,

    Below is the code I am trying, it fails at customerManager.create with exception "user is not authorised to perform this operation"

    private const string RetailServerSpn = "repetto.onmicrosoft.com/71e80ebd-af51-43c4-bc47-15f1fddca22b";;

           static public void Main(string[] args)

           {

               ManagerFactory managerFactory;

              //RetailServerContext context = RetailServerContext.Create(

              //new Uri(ConfigurationManager.AppSettings["RetailServerRoot"]),

              //ConfigurationManager.AppSettings["OperatingUnitNumber"]);

               AuthenticationContext authenticationContext = new AuthenticationContext(ConfigurationManager.AppSettings["aadAuthority"], false);

               AuthenticationResult authResult = null;

               authResult =  authenticationContext.AcquireTokenAsync(RetailServerSpn, new ClientCredential(ConfigurationManager.AppSettings["aadClientId"], ConfigurationManager.AppSettings["aadClientSecret"])).GetAwaiter().GetResult();

               ClientCredentialsToken clientCredentialsToken = new ClientCredentialsToken(authResult.AccessToken);

               RetailServerContext context = RetailServerContext.Create(new Uri(ConfigurationManager.AppSettings["RetailServerRoot"].ToString(), UriKind.Absolute), ConfigurationManager.AppSettings["operatingUnitNumber"], clientCredentialsToken);

               managerFactory = ManagerFactory.Create(context);

               // Getting a channelID.

               IOrgUnitManager orgUnitManager = managerFactory.GetManager();

               ChannelConfiguration channelConfiguration = orgUnitManager.GetOrgUnitConfiguration().GetAwaiter().GetResult();

               // Loading categories.

               try

               {

                   //ICategoryManager customerManager = managerFactory.GetManager();

                   //PagedResult categories = customerManager.GetCategories(channelConfiguration.RecordId,new QueryResultSettings { Paging = new PagingInfo { Skip = 0, Top = 100 } }).GetAwaiter().GetResult();

                   ICustomerManager customerManager = managerFactory.GetManager();

                   long salt = DateTime.Now.Ticks;

                   Customer customer = null;

                   customer = new Customer { AccountNumber = string.Empty, Email = "mail" + salt + "@contoso.com", FirstName = "Demo", CustomerTypeValue = 1 };

                   customer = customerManager.Create(customer).GetAwaiter().GetResult();

                   //PagedResult customers = customerManager.ReadAll(new QueryResultSettings { Paging = new PagingInfo { Skip = 0, Top = 100 } }).GetAwaiter().GetResult();

               }

               catch(Exception ex)

               {

               }

  • Community Member Profile Picture Community Member
    Posted 15 Sep 2017 at 14:27:39

    Hi Sergey,

    For me even customermanger.create as well doesn't work and gives the same error.

    My business case is all about intev6of customers products orders shipments with magento.

    So how do I pull all customer using ALI using retail proxy

  • SergeyP Profile Picture SergeyP 2,928
    Posted 13 Sep 2017 at 06:22:36

    Hi VKJ, CustomerManager.ReadAll is not implemented. If it would be implemented it would result in returning all customers (page by page) without any filter which doesn't have much sense from business point of view taking into account that there could be hundred of thousands of customers

  • Community Member Profile Picture Community Member
    Posted 07 Sep 2017 at 11:06:37

    Hi Sergey Pikhulya

    Great post...I have tried this...but when i try to  execute method ReadAll of customerManager , as below

    ICustomerManager customerManager = managerFactory.GetManager();

    PagedResult customers = customerManager.ReadAll(new QueryResultSettings { Paging = new PagingInfo { Skip = 0, Top = 100 } }).GetAwaiter().GetResult();

    I get following exception

    Exception of type 'Microsoft.Dynamics.Commerce.RetailProxy.UserAuthorizationException' was thrown.The user is not authorized to perform this operation.

    I do not get any error while fetching categories etc. Any idea why this error is coming?