logo
Forms
Last updated: Nov 26, 2024

How to get data from Microsoft Dynamics 365 Business Central and use it in public web forms

Co-Founder at Plumsail

In this article, I'll show you how to pull data from your Microsoft Dynamics 365 Business Central with the help of Azure Function and Entra ID app and use it in a public web form for populating fields and dropdown options. I will bind records from Customers, Vendors, and Items in Business Central to dropdown fields in a public web form designed with our Plumsail Forms and demonstrate how to dynamically populate other fields based on the selected option in the Customer dropdown—address, email, and others. Yet, with a few tweaks to the Azure Function, you can request any other data from your Business Central, limit the set of fields you want to expose, or configure filtration based on user's input as Microsoft provides comprehensive API for data manipulation.

Show data from Dynamics 365 Business Central in your public web form

 

 

What is Microsoft Dynamics 365 Business Central?

Dynamics 365 Business Central is an ERP system standing aside from other Dynamics 365 apps such as Sales, Customer Service, or Talent which use Dataverse for storing their data and thus are integrated into Microsoft Power Platform. Business Central is a standalone solution which nevertheless can be connected to other Dynamics 365 apps via REST API. It's designed for small and mid-sized organizations for managing their finances, sales, procurements, and products.

You can sign up for a trial with your Microsoft work or school account but unfortunately, in most of the countries the trial is unavailable. You can either find a Microsoft reselling partner or create a new tenant in Microsoft Entra ID and select the US as datacenter location as the region cannot be changed for existing tenants.

 

Register an app with Microsoft identity platform

First, we need to register a new Entra ID app with access to Dynamics 365 Business Central API.

Navigate to App registrations in Entra admin center

  • Click New registration and give your app some name, Azure Functions (Demo) in my case
  • Leave the Supported account types on the default setting of Accounts in this organizational directory only
  • Click Register

Register an Entra ID app

  • In Overview, copy the Application (client) ID of your app and save it somewhere. We will use it in Azure Function.

Copy Application (client) ID of your Entra ID app

Now, we need to configure permissions for the app:

  • Navigate to API Permissions and click Add a permission
  • Under the Microsoft APIs tab, select Dynamics 365 Business Central
  • Choose Delegated permissions. I have also tried to get data using Application permissions or Microsoft Graph API (beta) but constantly received Authentication_InvalidCredentials error. Perhaps, it was a temporary issue but Delegated permissions did work.
  • Check Financials.ReadWrite.All
  • Click Add permissions

Finally, we need to create a client secret for our app to use it in Azure Function:

  • Navigate to Certificate & secrets and click New client secret
  • Enter any description, specify lifetime, click Add
  • Save the created secret key somewhere.

 

Deploy an Azure Function

You can deploy functions for retrieving data from Dynamics 365 Business Central directly from our GitHub repository. No need to modify source code to specify the app or the data source settings, all of them can be configured via the Function App's environment variables after the deployment:

  • Create a Function App in your Azure portal.
  • Select .NET as Runtime stack and 8 (LTS) isolated as its version.

Create Azure Function app

Fork GitHub repository

  • Return to the Function App and navigate to Deployment Deployment center:
  • In the Source, select GitHub and sign into your account
  • Select your fork of the data-source-functions repository

Configure repository for your Azure Function

  • Click Save

Once the project is built and deployed, you will see the functions in your Function App. Now, we need to define its environment variables to get access to our Dynamics 365 Business Central.

  • Open the Function App and navigate to Settings Environment variables
  • Add the following properties to the App settings list:
Dynamics365.BusinessCentral:AzureApp:ClientId
The Application (client) ID of the Entra ID app

Dynamics365.BusinessCentral:AzureApp:ClientSecret
The Client secret of the Entra ID app

Dynamics365.BusinessCentral:AzureApp:Tenant
Your Microsoft 365 tenant, ex.: contoso.onmicrosoft.com

Dynamics365.BusinessCentral:AzureApp:InstanceId
Your Business Central ID. Take it from your Business Central URL:
https://businesscentral.dynamics.com/{instanceId}/?referrer=office

Dynamics365.BusinessCentral:Customers:Company
The name of the company in Dynamics 365 Business Central which Customers you want to request in D365-BC-Customers function

Dynamics365.BusinessCentral:Vendors:Company
The name of the company in Dynamics 365 Business Central which Vendors you want to request in D365-BC-Vendors function

Dynamics365.BusinessCentral:Items:Company
The name of the company in Dynamics 365 Business Central which Items you want to request in D365-BC-Items function

Set up environment variables for Azure Function

Finally, we need to enable Cross-Origin Resource Sharing (CORS) for our app to allow JavaScript requests from our web form:

  • Open API CORS
  • Add “*” to Allow Origins and remove all other values

Configure CORS for Azure Function

 

Authorize the app

You might noticed the D365-BC-Authorize function among the others. It allows you to grant delegated permissions to the app. All further communication with Business Central will be performed on behalf of a user who approved the app's permission request. After approval of the app request, the Azure Function saves tokens in cache and uses them in other functions for obtaining data from Business Central.

Before starting this function, you should add its URL to the list of redirect URLs of the app.

  • Open Overview D365-BC-Authorize function:
  • Click Get function URL and copy its URL:

Copy Azure Function URL

  • Open the Entra ID app we've created at the beginning (Identity App registrations your app) and navigate to Authentication section
  • Click Add platform, select Web, and insert the function URL into Redirect URIs text box
  • Click Configure

Specify Redirect URI for Entra ID app

  • Insert the function URL into your browser
  • Sign in as a user with the permissions to Dynamics 365 Business Central
  • Accept the requested permissions:

Approve permission request from Entra ID app

 

Populate dropdown options

Finally, we can design a public web form and populate dropdown field options from Azure Functions with JavaScript. In this example, I have Commercial proposal form with Customer, Vendor, and Items fields which I want to be linked to corresponding lists in Dynamics 365 Business Central.

First, I need to copy their names in the design mode:

Copy field names in Plumsail Forms designer

Next, copy URLs of D365-BC-Customers, D365-BC-Vendors, and D365-BC-Items functions and remove {id} from them. Should be like that:

https://{your-function-app}.azurewebsites.net/api/bc/customers/?code=...

{id} is needed if we want to retrieve a certain Customer, Vendor, or Item. Without specifying {id}, the functions return all records.

Switch to JavaScript editor and assign functions' URLs as data sources for the dropdown fields:

fd.rendered(() => {
    const customerField = fd.field('Customer');
    const customerWidget = customerField.widget;
    customerWidget.setDataSource({
            transport: {
                read: '-- URL of D365-BC-Customers (without {id}) --'
            }          
        });
    customerWidget.setOptions({dataTextField: 'displayName', dataValueField: 'id'});

    const vendorsWidget = fd.field('Vendors').widget;
    vendorsWidget.setDataSource({
            transport: {
                read: '-- URL of D365-BC-Vendors (without {id}) --'
            }          
        });
    vendorsWidget.setOptions({dataTextField: 'displayName', dataValueField: 'displayName'});
    
    const itemsWidget = fd.field('Items').widget;
    itemsWidget.setDataSource({
            transport: {
                read: '-- URL of D365-BC-Items (without {id}) --'
            }          
        });
    itemsWidget.setOptions({dataTextField: 'displayName', dataValueField: 'displayName'});
});

And here you go:

Populate droprowns with data from Dynamics 365 Business Central

 

Populate fields based on the selection

Now, once a user selects a Customer, I want to populate Customer Info section with properties of the selected Customer—address, email, and website. Switch to JavaScript editor and add the following function for requesting a customer by id and populating the customer info fields:

async function populateCustomerData(customerId) {
    // Replace {id} in D365-BC-Customers URL with customerId
    const response = await fetch(`https://{your-function-app}.azurewebsites.net/api/bc/customers/${customerId}?code=...`);
    if (response.ok) {
        const customer = await response.json();
        fd.field('Address1').value = customer.addressLine1;
        fd.field('Address2').value = customer.addressLine2;
        fd.field('City').value = customer.city;
        fd.field('Country').value = customer.country;
        fd.field('State').value = customer.state;
        fd.field('ZIPCode').value = customer.postalCode;
        fd.field('Email').value = customer.email;
        fd.field('Website').value = customer.website;
    } else {
        fd.field('Address1').value = 
            fd.field('Address2').value = 
            fd.field('City').value = 
            fd.field('Country').value = 
            fd.field('State').value = 
            fd.field('ZIPCode').value = 
            fd.field('Email').value = 
            fd.field('Website').value = null;
    }
}

Here, we replace {id} from the D365-BC-Customers URL with the id of a particular customer. Now, we'll need to call this function whenever a user changes selection in the Customer field:

fd.rendered(() => {
    const customerField = fd.field('Customer');
    customerField.$on('change', v => populateCustomerData(v));
});

And this is my public web form populated with data from Business Central: Populate public web form with data from Dynamics 365 Business Central

 

Conclusion

This was just a basic example of what can be achieved with Azure Functions and Plumsail forms. The code is not optimal and designed only for demonstrating capabilities. You're not limited to just requesting data but also, you can submit data back to your Azure Function and save it to Dynamics 365 Business Central or any other app. In Plumsail Forms, you can add a URL of your function to Webhooks in Settings Notifications and each time a user submits your form, the data will be sent to the specified URL.

If you have any questions, do not hesitate to contact us by email or in Plumsail community. And let us know if you are looking for integration of Plumsail forms with other platforms, we'll be happy to demonstrate it in the future blog posts. All the best and stay tuned!