Tuesday, March 31, 2015

Dynamic AX Query Services in BizTalk Functoid:

Author: Toraj Khavari
Date: March. 31, 2015

Objective: You may at times face querying Microsoft Dynamics AX (AX) data for BizTalk mapping. There are a few methods to implement BizTalk solutions to query AX data in BizTalk. This article is written to implement a C# in-line code, using BizTalk Script Functoid with signed assembly to Query AX data, and employing Dynamic AX Services. The assembly is Global Assembly Cache (GAC) registered and environments configurable.

AX has a powerful Application Interface Frame (AIF). Querying data from AX is very straight forward. For an introduction to AX Query, please refer to the following articles.

Article Body:

1-      Create a C# class library project, add a service reference for the DynamicsAx/Services/QueryService, sign the library project, and version the project’s assembly. Versioning the project assembly will assist you and the production support staff. For more details about assembly version, refer to the following article.
How to Version Solution Assemblies and why should we do them - http://torajkhavari.blogspot.com/2015/03/how-to-version-solution-assemblies-and.html

In my case, the business requirements were to get the following components from the AX’s “PurchLine” table. The query requires Purchase Order or PurchID, ItemID and Quantity.
  • InventDimId
  • InventTransRefId
  • InventTransId

Project class name space is: Adec.BTSUtility.PurchLineMetaDataQueryService
Class name is: AOTPurchLineMetaDataQuery
Method name is: GetInventDimIdInventTransRefIdInventTransId

To save runtime, the method will return all three values with special characters delimiters. BizTalk can extract the three values during runtime keying on special characters.

Copy and paste the following code in the project “GetInventDimIdInventTransRefIdInventTransId” method.

#region GetInventDimIdAndInventTransRefId
/// <summary>
/// Gets the invent dim identifier, invent trans reference identifier and invent trans identifier.
/// </summary>
/// <param name="po">The Purchase Order.</param>
/// <param name="itemID">The item identifier.</param>
/// <param name="quanity">The quantity.</param>
/// <returns>String InventDimId followed by a marker, InventTransRefId followed by a marker, and InventTransId followed by a marker</returns>
public string GetInventDimIdInventTransRefIdInventTransId(string po, string itemID, string quantity)
{
    // Refrences:
    // http://blogs.msdn.com/b/axsupport/archive/2012/02/17/calling-the-query-service.aspx
    // https://team.a-dec.com/sites/SolDev/Developer%20Wiki/Dynamics%20AX%20An%20Introduction%20to%20Query%20Services.aspx
    // http://ax2012aifintegration.blogspot.com/2012_05_01_archive.html
    string strResult, tempStrInventDimId, tempStrInventTransRefId, tempStrInventTransId, eventLogStr = null;

    /// Eventlog status
    eventLogStr = System.String.Format("Start Invent Query for PO={0}, ItemID={1}, Quanity={2}. In memory purchaseOrder is {3}. ",
                                                        po, itemID, quantity, purchaseOrder);
    eventLogStr = eventLogStr + System.String.Format("Class {0} a member of namespace {1}", "GetInventDimIdAndInventTransRefId", "Adec.BTSUtility.PurchLineMetaDataQueryService");
    System.Diagnostics.EventLog.WriteEntry("VendorsEDI", eventLogStr, System.Diagnostics.EventLogEntryType.Information, 1000);
    eventLogStr = "";


    /// always catch exception and error. Report problem.
    try
    {
        /// Always save the latest purchase order that is passed.
        if (System.String.IsNullOrEmpty(purchaseOrder))
        {
            purchaseOrder = po;
        }

        if (System.String.IsNullOrEmpty(po) == false)
        {
            if (po.Length > 0)
            {
                purchaseOrder = po;
            }
        }

        if (quantity.Contains(".00") == false)
        {
            quantity = quantity + ".00";
        }

        // Instantiate Query Client, Query Meta data, and Query Data Source for Meta Data
        QueryServiceClient client = new QueryServiceClient();
        AxQueryServiceReference.QueryMetadata query = new AxQueryServiceReference.QueryMetadata();
        query.DataSources = new AxQueryServiceReference.QueryDataSourceMetadata[1];

        // Identify the Query
        query.Name = "PurchLine";

        // Instantiate Query Data Source for Meta data
        AxQueryServiceReference.QueryDataSourceMetadata queryDataSourceMD = new AxQueryServiceReference.QueryDataSourceMetadata();

        // Identify the query name and its table
        queryDataSourceMD.Name = "PurchLine";
        queryDataSourceMD.Table = "PurchLine";
        queryDataSourceMD.Enabled = true;

        // Identify the ranges.
        AxQueryServiceReference.QueryDataRangeMetadata rangePO = new AxQueryServiceReference.QueryDataRangeMetadata();
        rangePO.Enabled = true;
        rangePO.TableName = "PurchLine";
        rangePO.FieldName = "PurchId";
        rangePO.Value = purchaseOrder;
        rangePO.Enabled = true;

        AxQueryServiceReference.QueryDataRangeMetadata rangeItemID = new AxQueryServiceReference.QueryDataRangeMetadata();
        rangeItemID.TableName = "PurchLine";
        rangeItemID.FieldName = "ItemId";
        rangeItemID.Value = itemID;
        rangeItemID.Enabled = true;

        AxQueryServiceReference.QueryDataRangeMetadata rangeQuanity = new AxQueryServiceReference.QueryDataRangeMetadata();
        rangeQuanity.TableName = "PurchLine";
        rangeQuanity.FieldName = "QtyOrdered";
        rangeQuanity.Value = quantity;
        rangeQuanity.Enabled = true;

        // Set the ranges
        queryDataSourceMD.Ranges = new AxQueryServiceReference.QueryRangeMetadata[3];
        queryDataSourceMD.Ranges[0] = rangePO;
        queryDataSourceMD.Ranges[1] = rangeItemID;
        queryDataSourceMD.Ranges[2] = rangeQuanity;
               
        query.DataSources[0] = queryDataSourceMD;

        queryDataSourceMD.DynamicFieldList = true;


        // Get all the records
        Paging paging = null;

        DataSet dataset = client.ExecuteQuery(query, ref paging);

        strResult = "";
        // Always return the 1st item found
        if (dataset.Tables[0].Rows.Count > 0)
        {
            DataRow datarow = dataset.Tables[0].Rows[0];
            tempStrInventDimId = datarow.Field<string>("InventDimId"); // Use this for BizTalk WMSJournalTrans.InventDimId
            tempStrInventTransRefId = datarow.Field<string>("PurchID"); // Use this value for BizTalk Map InventTransRefId
            tempStrInventTransId = datarow.Field<string>("InventTransId"); // Use this for BizTalk WMSJournalTrans.InventTransId

            // "<" Begining of the InventTransRefId, ">" for the beginning of StrInventTransRefId, and ";" at the end of InventTransId.
            strResult = strResult + string.Format("{0}<{1}>{2};", tempStrInventDimId, tempStrInventTransRefId, tempStrInventTransId);
        }
        else
        {
            strResult = strResult + string.Format("No Match for PO={0}, ItemID={1}, Quanity={2}, in class {3}",
                                                    purchaseOrder, itemID, quantity, "GetInventDimIdAndInventTransRefId");

            System.Diagnostics.EventLog.WriteEntry("VendorsEDI", strResult, System.Diagnostics.EventLogEntryType.Warning, 1001);
            return strResult;
        }
    }
    catch (Exception e)
    {
        strResult = string.Format("Exception thrown in the {0} method. A member of {1} class in the {2} namespace. ", "GetInventDimIdAndInventTransRefId",
                                    "AOTPurchLineMetaDataQuery", "Adec.BTSUtility.PurchLineMetaDataQueryService");
        strResult = strResult + string.Format("Error Message:{0}\n", e.Message);
               
        // Log exception
        System.Diagnostics.EventLog.WriteEntry("VendorsEDI", strResult, System.Diagnostics.EventLogEntryType.Error, 1009);
    }

    // Log complete
    eventLogStr = System.String.Format("End Invent Query for PO={0}, ItemID={1}, Quanity={2}. In memory purchaseOrder is {3}. ",
                                            po, itemID, quantity, purchaseOrder);
    eventLogStr = eventLogStr + System.String.Format("Class {0} a member of namespace {1}", "GetInventDimIdAndInventTransRefId", "Adec.BTSUtility.PurchLineMetaDataQueryService");
    System.Diagnostics.EventLog.WriteEntry("VendorsEDI", eventLogStr, System.Diagnostics.EventLogEntryType.Information, 1000);
           
    return strResult;
}
#endregion GetInventDimIdAndInventTransRefId

2-      Copy the Adec.BTSUtility.PurchLineMetaDataQueryService.dll and its configuration file to the BizTalk server. In my case, I copied them in the BizTalk Server C:\Adec\Adec.BTSUtility.PurchLineMetaDataQueryService.

3-      In the BizTalk server Click Start, Type cmd, or click All Programs, and then click Accessories. Right-click Command prompt, and then click Run as administrator. If the User Account Control dialog box appears, confirm that the action it displays is what you want, and then click Continue.

4-      Navigate to the gacutil.exe folder. In my case as follows.
cd C:\Program Files (x86)\Microsoft SDKs\Windows\v8.0A\bin\NETFX 4.0 Tools

5-      Register the dll with GAC.
gacutil.exe /if "C:\Adec\Adec.BTSUtility.PurchLineMetaDataQueryService\Adec.BTSUtility.PurchLineMetaDataQueryService.dll"

6-      Verify the GAC registration. In A-dec server, its location as follows.
GAC Location:\Windows\Microsoft.NET\assembly\GAC_MSIL\Adec.BTSUtility.PurchLineMetaDataQueryService
In the server’s GAC, the dll’s assembly version number will be very handy to validate the latest registration.

7-      Although BizTalk does not use the dll’s configuration file, I recommend copying the Adec.BTSUtility.PurchLineMetaDataQueryService.dll.config next to its assembly in GAC. It is for future traceability.

8-      Modify the BTSNTSvc.exe.config or BTSNTSvc64.exe.config depending if it runs on 32 or 64 bits process to contain the  Adec.BTSUtility.PurchLineMetaDataQueryService.dll.config contents. In the BizTalk server in A-dec, they are located in the following folder.
C:\Program Files (x86)\Microsoft BizTalk Server 2013

Copy the following information in the BTSNTSvc.exe.config or BTSNTSvc64.exe.config.

<!-- app.config for Adec.BTSUtility.PurchLineMetaDataQueryService class -->
<system.serviceModel>
      <bindings>
                  <netTcpBinding>
                              <binding name="QueryServiceEndpoint" transferMode="Streamed" maxReceivedMessageSize="2147483647" />
                  </netTcpBinding>
      </bindings>
      <client>
      <endpoint address="net.tcp:// YourAXAOSServer:port/DynamicsAx/Services/QueryService"
                  binding="netTcpBinding" bindingConfiguration="QueryServiceEndpoint"
                  contract="AxQueryServiceReference.IQueryService" name="QueryServiceEndpoint">
        <identity>
                  <userPrincipalName value="YourAXServiceId" />
        </identity>
      </endpoint>
      </client>
</system.serviceModel>

9-      At this point BizTalk configuration is completed.

10- In the Visual Studio’s map project add the GAC Adec.BTSUtility.PurchLineMetaDataQueryService.dll’s reference.

11- In the MAP add a script Functoid with Select script type, “External Assembly”. Navigate to the Script assembly, class and method. The assembly requires three inputs and has one output. Connect the input and output to desired schema node.


12- Rebuild and deploy. The application is ready for end to end testing.

Hint:
  • The Visual Studio Test Map will not work with external assemblies. Except, if the Visual Studio config file, devenv.exe.config, is updated with Query Service configuration information.
Aline your gifts, talents, education, and work, you will feel and know success, happiness and contentment. Happy coding.
Cheers Toraj


Version 1.0.0331.01