Friday, October 1, 2021

Dynamics 365 Field service

Field service

Important terminologies:

1. Work orders: Gives info abt work that needs to be done

    - Can be created manually

    - Converted from an opp

    - Converted from a case

    - Created from an agreement

2. Agreements: Allows generating work orders and invoices automatically

3. Purchase orders: allow adding inventories to the warehouse

4. Inventories transfer: transfer product from 1 warehouse to another.

5. RMA: Return Merchandise Authorization: Created when product is returned

6. Scheduled Board: allow scheduling resources & track the work order activities

Other terms are the same as the Sales Module.

Process flow of field service:

Opportunity/case->work order created -> resource scheduling -> resource mob. Service delivery->invoicing -> RMA(if any)

Dynamics 365 Project Service

Project Service

Important terminology:

    - Project Contract
    - Project template

    - Projects

    - Invoices 

    - Resources 

    - Time entries

    - Expenses

    - Schedule board

Other terms are the same as the Sales Module.

Project service process flow:

Opportunity/quote -> project planning -> project contracts-> resourcing->record time/expense -> invoicing-> send to the customer for payment

Dynamics 365 Customer Service

Customer service
 

Important terminology:

1. Cases: It allows companies to resolve issues faced by their customers.
    - Create a case 

    - Assign a service agent 

    - Resolve the case - communicate to customer abt resolution

    - Reopen if required 

2. Service calendar: It's a place where all the service activities scheduled can be seen.

    - Resource wise service scheduling

    - Shows activities with statuses

    - Synchronized with outlook

3. Queues: Used to organize, prioritize and monitor the progress of work

    - Public Q - viewed by everyone - Private Q - specific issue 

    - May be used for different levels of cases - high, low

    - Queue items are created

    - User can focus on completing tasks assigned to him/her

4. Articles: An article can be used as a knowledge base for service agent

    - Pdt guide

    - Datasheets

    - Frequently Asked Questions

    - Problem solution

    - Standard operating procedures

5. Contracts: used to define accurate terms and condition for the support to customers

    - No. Of cases

    - Times

    - Coverage Dates

    - A contract uses contract template

6. Services: It is a work performed for a customer (further same as sales )

    - Service duration

    - Define resource required

    - Define site required

    - Used by service scheduling engine

Other terms are the same as the sales module.

Service process flow:

Issue logged (Question or feedback)-> Assign an agent -> Ask more info-> resolve issue -> Communicated (to customer)-> reopen case(if any issue)-> proceed to step 2 

Dynamics 365 Marketing Overview

Marketing: 

Important terminology:

1. Marketing Lists: This allows you to the group account, contact, leads for marketing
    - Choose either account or contact or lead

    - 2 types - Dynamic & Static

2. Campaign: This allows u to plan & execute a marketing project

    - Define campaign details

    - Choose marketing list

    - Create & run campaign activity

Other terms are the same as Sales Module.

Marketing Process Flow:

Plan campaign -> attach marketing list-> marketing activity-> define budget->Execute the campaign (create activity)-> generate lead-> follow the sales process 

Thursday, September 30, 2021

Overview of Dynamics 365 Sales Module


  • Introduction:

Microsoft Dynamics 365 is designed to support the sales process from acquiring a new lead through the close of a sale. CRM has a place to store the contact information for new leads, a place to track the follow-up communications (such as Phone Calls, Emails, and Appointments), and the ability to qualify a Lead into an Account, Contact, and Opportunity.

  • Terminologies:

  1. Account: Organization; Includes Customer, Vendor, Partner, Affiliate, or Other.

  2. Contacts: Individual; Associated with a maximum of one account (through the contact form).

  3. Leads: Prospect (Potential Customer/ Potential Sale).

  4. Opportunities: Potential Sale (Created when a lead is almost ready to buy a product or service).

  5. Quote: Document for customer/prospect; Contains information about Product, Quantity, Pricing, Payment Terms, and other important details.

  6. Order: Confirmation from customer/ prospect on buying a product.

  7. Invoices: Document containing billing information.

  8. Competitors: Information about competitors. So that you can keep track of them and win.

  9. Sales literature: Centralized repository for sales-related documents. Contains Brochures, Product Guides, Competitors information, Pricing and Discounts, Sales documents, etc.

  10. Product Catalog: Collection of Products and their pricing information.

  11. Goals: Used to keep track of progress on achieving target revenue.

  12. Goal Metrics: Explains how the goal number or figure I measured.

  • Process Flow:

Step 1: Capture Lead - Create Lead.

Step 2: Account Creation - Information captured of a company or company details become Account after lead qualification process.

Step 3: Contact Setup - Information captured of an individual in a lead becomes Contact after the lead qualification process.

Step 4: Opportunity management - Once the Lead shows interest in a product and asks for more information Qualify the lead. Now the Lead becomes an Opportunity.

Step 5: Product Catalog - Add all the information about products and their pricing.

Step 6: Quote Management - Create Sales Quote.

Step 7: Order Management - Once the customer accepts everything mentioned in Sales Quote and confirms, convert the quote to order.

Step 8: Close Opportunity.

Step 9: Invoice Management - Create a Sales Invoice and send it to the customer.

Step 10: Sales Business Process - Businesses can define a process flow so that sales users will follow the guided process without any confusion.

Image


References: https://community.dynamics.com/crm/b/sandeepmishracrmblog/posts/microsoft-dynamics-365-sales-module-overview


Thank you!!!
Happy Leaning...


Dynamics 365 Plugin Event Execution Pipeline

The Microsoft Dynamics CRM event processing subsystem executes plug-ins based on a message pipeline execution model. 

The message contains business entity information and core operation information. 

The message is passed through the event execution pipeline where it can be read or modified by the platform core operation and any registered plug-ins.

Note: While there are several Web services hosted by the Microsoft Dynamics CRM platform, only events triggered by the organization and OData endpoints can cause plug-ins to execute.


Plugins can be triggered by:
A user action in the Dynamics 365 user interface, such as retrieving a record
Events triggered by the Organization Service or the WebAPI, such as retrieving a record

Post-Operation events in the pipeline are processed either:
Synchronously – these plugins are executed immediately and run in a specific order
Asynchronously – these plugins are executed by the Queue Agent and run later by the async service Other events are processed synchronously.

Event Execution Pipeline: 

Pre-event > pre - validation - Stage 10:

        1. Executes before the main system operation
        2. Plugin register in this stage execute outside the database transaction
        3. Pre validation stage occurs prior to a security check being performed to verify the calling or logged-in user has the correct permission to perform the intended operation.
        4. The operation which is performed in this stage can't be rolled back even if it failed(till the code execution).

Pre-event > Pre-operation - Stage 20:

1. Executes before the main system operation.
2. Plugin registered in this stage executes within the database transaction
        3. The operation which is performed in this stage will be rolled back if it's failed, even if one task is successful it will roll back completely.

Main operation - Platform core operation - Stage 30:

1. In transaction - main operation of the system such as - create, update delete and so on. 
2. No custom plugin can be registered
        3. For internal use only.

Post-event > Post operation - Stage 40:

1. Plugin registered in this stage executes after the main operation.
2. Within the database transaction
        3. The operation which is performed in this stage will be rolled back if it's failed, even if one task is successful it will roll back completely.

Images:

Pre-image:

A pre-image is a snapshot of entity attributes before the main operation.

Stage     create   update    delete

Pre          NO         Yes           Yes

Post        Yes         Yes           Yes

Post-Image:

A post image is a snapshot of the entity's attribute after the core operation

Pre         No          No            No

Post       Yes          Yes          No

The best way, When to use which pipeline event to register plugin :

1. Pre-Validation (outside the database transaction): avoid Update, Create, or Delete in this stage. This stage is good for the validation process (eg: throw an error if the position of the user is not high enough).

2. Pre-Operation (inside the database transaction): processes that run before the record being saved on the database. If got an error this stage of after this stage, the data will be rollback.

3. Main-Operation: we can’t custom in here. Dynamics CRM main process to save the record in the database.

4. Post Operation (inside the database transaction): the process that can be put after the record being saved on the database. 

Note: As suggested, we should only use Pre-Operation and Post-Operation. The reason is Pre-Operation and Post-Operation are the only stages that support roll-back. While Pre-Validation is not transactional.


Thank you!
Happy Learning...

Saturday, July 31, 2021

Custom Workflow activity to get and set partylist lookup field using C# code

Follow the step-by-step process to Get the value from the email activity From field and set the value in the customer type lookup field.

In my scenario, I want to Get the value from the email activity From partylist field and Set the value in the custom field which is of Customer Type. (i.e. From & Customer)

 Step by step follow below custom activity code:

// Declare InArgument & OutArgument

//Pass email input argument to workflow
        [RequiredArgument]
        [Input("Email")]
        [ReferenceTarget("Email")]
        public InArgument<EntityReference> Email { get; set; }

//Return output parameter which is of contact type in my case
        [ReferenceTarget("contact")]
        [Output("customer")]
        public OutArgument<EntityReference> customer { get; set; }


Write below code inside execute method:


var email = Email.Get(context);
var emailDetails = service.Retrieve(email.LogicalName, email.Id, new ColumnSet("from"));
                    if (emailDetails.Contains("from") && emailDetails["from"] != null)
                    {
                        EntityCollection FromCollection = emailDetails.GetAttributeValue<EntityCollection>("from");


//To get the from reference and set on the Email customer lookup

                        EntityReference fromContact = GetFromContact(FromCollection);
                        customer.Set(context, fromContact);
                    }

//Method to get the value from From field and Set it to Customer Lookup

public EntityReference GetFromContact(EntityCollection fromCollection)
        {
            EntityReference contact = null;
            try
            {
                if (fromCollection != null && fromCollection.Entities.Count > 0)
                {
                    var from = fromCollection.Entities.First();

//To get the entity type
                    var partyId = from.GetAttributeValue<EntityReference>("partyid");
                    tracingService.Trace("PartyId: " + partyId.Id + " PartyId LogicalName: " + partyId.LogicalName);
                    if (partyId.LogicalName == "contact")
                    {

//To set the customer lookup
                        contact = new EntityReference(partyId.LogicalName, partyId.Id);
                    }
 
                }
            }
            catch (Exception ex)
            {
            }
            return contact;

        }


Build, Sign and register the workflow activity code using the Plugin registration tool

Create an OOB Workflow Activity and add the input variable and set the output variable to update.


Thank you..

Keep Learning..!!

Friday, July 30, 2021

Get and Set activityparty entity and Partylist field using C# code in D365

This blog will give you overview on activity Party entity and partylist fields.

Within Dynamics 365 we have activity party entity as well as partylist fields, an activity party represents a person or group associated with an activity. An activity can have multiple activity parties. 
There are 11 activity party types in Dynamics 365 for Customer Engagement (Sender "From", ToRecipient, CCRecipient, BccRecipient, RequiredAttendee, OptionalAttendee, Organizer, Regarding, Owner, Resource, Customer).

CRM Activity PartyList field is a special type of lookup field which refers to multiple entities. To hold or store these multiple entities references, Microsoft has provided one special type of entity named ActivityParty.

ActivityParty, has one special field called PartyId. which has the capability to store multiple entities references like user, contact, account, lead, queue etc.

Hence, if you want to get/set value in PartyList Field, you need to interact with ActivityParty PartyId field.

Remember, while getting or setting value in party list fields (to, from cc, bcc, regarding etc). You should use ActivityParty Entity and PartyId field. 

1. To get the value from Party List TO and FROM fields:

Retrieve the value from PartyId field of ActivityParty entity, which holds the record’s Guids which was selected in Email’s FROM and TO fields.

2. To set the value in Party List TO and FROM fields:

First pass the value in PartyId field of ActivityParty entity and then set it’s EntityReference to partylist FROM and TO fields.

Note: Use the dynamics value in place of hardcoded values.

Get partyList fields value using C# code:

public static void GetPartyList(string emailRecordId)
        {
            emailRecordId = "C472398F-592F-E561-A939-000S4AF24369";
            // Retrieve email record
            Entity email = service.Retrieve("email", new Guid(emailRecordId), new ColumnSet("from", "to"));
 
            // get value from partylist - 'FROM' field
            EntityCollection from = email.GetAttributeValue<EntityCollection>("from");
            if (from != null && from.Entities.Count > 0)
            {
                foreach (var item in from.Entities)
                {
                    EntityReference partyId = item.GetAttributeValue<EntityReference>("partyid");
                    string addressUsed = item.GetAttributeValue<string>("addressused");
                }
            }
 
            // get value from partylist - 'TO' field
            EntityCollection to = email.GetAttributeValue<EntityCollection>("to");
            if (to != null && to.Entities.Count > 0)
            {
                foreach (var item in to.Entities)
                {
                    EntityReference partyId = item.GetAttributeValue<EntityReference>("partyid");
                    string addressUsed = item.GetAttributeValue<string>("addressused");
                }
            }

        }

//-----------------------------------------------------------------------------------------------

Set value in Party List Fields using C# (Single Recipient)

//Set partyList field when there is single recipient
        public static void SetPartyList()
        {
            // Declare party list entity to set value in FROM & TO field
            Entity from = new Entity("activityparty");
            Entity to = new Entity("activityparty");
 
            // Set value in FROM & TO field
            from["partyid"] = new EntityReference("systemuser", new Guid("D7241D9E-FCEC-4C6B-80D0-A2C5B1588D5"));
            to["partyid"] = new EntityReference("account", new Guid("475B158C-4C6C-E511-8C63-72D8D9E47BA8"));
 
            // Declare party list entity to set value in FROM field
            Entity email = new Entity("email");
 
            // Insert value in email FROM & TO field
            email["from"] = new Entity[] { from };
            email["to"] = new Entity[] { to };
 
            //Set regarding object property (i.e. The entity record, which u want this email associated with)
            EntityReference regardingObject = new EntityReference("contact", new Guid("6C3BE98B-88DF-E311-B8E5-6HGE55B8B676"));
 
            email.Attributes.Add("regardingobjectid", regardingObject);
 
            //Set subject & body properties
            email.Attributes.Add("subject", "Email subject created for Single recipients");
            email.Attributes.Add("description", "This is Email description for Single recipients");
 
            //Create email activity
            Guid emailID = service.Create(email);

        }

Set value in Party List Fields using C# (Multiple Recipient)

//Set partyList field when there is Multiple recipient

        public static void SetPartyList()
        {
            // Set value in FROM party list field
            Entity from1 = new Entity("activityparty");
 
            // two accounts inside the TO field
            Entity to1 = new Entity("activityparty");
            Entity to2 = new Entity("activityparty");
            // two contacts inside the TO field
            Entity to3 = new Entity("activityparty");
            Entity to4 = new Entity("activityparty");
 
            // set value in FROM field
            from1["partyid"] = new EntityReference("systemuser", new Guid("D72D8D9E-FCEC-4C6B-8340-A2CB9FAA88D5"));
 
            // set multiple values in TO field
            to1["partyid"] = new EntityReference("account", new Guid("475B158C-541C-E511-80D3-3863BB347BA8"));
            to2["partyid"] = new EntityReference("account", new Guid("A8A19CDD-88DF-E311-B8E5-6C3BE5A8B200"));
            to3["partyid"] = new EntityReference("contact", new Guid("25A17064-1AE7-E611-80F4-E0071B661F01"));
            to4["partyid"] = new EntityReference("contact", new Guid("48A0E5B9-88DF-E311-B8E5-6C3BE5A8B980"));
            Entity email = new Entity("email");
 
            //Set regarding object property (i.e. The entity record, which u want this email associated with)
            EntityReference regardingObject = new EntityReference("contact", new Guid("48A0E9B9-88DF-E311-B8E5-6C3BE5A8B980"));
 
            email.Attributes.Add("regardingobjectid", regardingObject);
 
            // Insert value in FROM & TO field
            email["from"] = new Entity[] { from1 };
            email["to"] = new Entity[] { to1, to2, to3, to4 };
 
            //Set subject & body properties
            email.Attributes.Add("subject", "Email created from Console for Multiple recipients");
            email.Attributes.Add("description", "Email created from Console or Multiple recipients");
 
            //Create email activity
            Guid emailID = service.Create(email);

        }


References: Click HereMS Docs ActivityParty

Thank you!!!

Keep CRMing... 

Happy Learning.....

Thursday, July 8, 2021

Plugin code to update associated record in D365

How to update multiple associated records using D365 plugin code?

Here in this scenario, I have a custom entity called "Sample" and it's having a 1:N relationship with a Case entity. Now I want to update few fields on the "Sample" record and it should automatically update few fields on the Case record once click on the save record.

Let's proceed... 

1. Get the Sample record Id first which can be passed to fetch XML conditions to update the only related record to this ID.

2. Follow the code below, fetch the case record using advance find, convert it to C# format and add it variable as shown
     -> Pass the formated fetch XML to service.retrieveMultiople to fetch the record collection
    -> Add it to an entity collection and read and update records one by one
    -> Call the service.update with entity object in it.

Entity esample;

DateTime createdon = esample.GetAttributeValue<DateTime>(Sample.Createdon);//"createdon"
DateTime modifiedon = esample.GetAttributeValue<DateTime>(Sample.Modifiedon);//"modifiedon"
            try
            {
               
                var fetchXml = @"<fetch version='1.0' output-format='xml-platform' mapping='logical' distinct='false'>
                                  <entity name='incident'>
                                    <attribute name='incidentid' />
                                    <attribute name='createdon' />
                                    <attribute name='modifiedon' />
                                    <order attribute='createdon' descending='false' />
                                    <filter type='and'>
                                      <condition attribute='new_sampleid' operator='eq' value='{0}' />
                                    </filter>
                                  </entity>
                                </fetch>";
                fetchXml = string.Format(fetchXml, esample.Id);//Pass sample record id index
                caseResult = service.RetrieveMultiple(new FetchExpression(fetchXml));
                if (caseResult != null && caseResult.Entities.Count > 0)
                {
                    foreach (var caseRecord in caseResult.Entities)
                    {
                        Entity eCase = new Entity(Cases.EntityLogicalName);//"incident" entity logical name
                        eCase.Id = caseRecord.Id;
                        eCase[Cases.CreatedonDatetime] = createdon;//eCase["createdon"]
                        eCase[Cases.ModifiedonDatetime] = modifiedon;//eCase["modifiedon"]
                        service.Update(eCase);
                    }
                }

            } 


The record gets updated go and verifies the associated record.

Click here to learn more

Thats' all

Keep CRMing...

Wednesday, July 7, 2021

Retrieve label name of optionset value from link-entity fetch XML D365 C# Code

Follow the below process step by step to get the label name from the option set attribute from the linked entity.

Country and city are two related entities. In the city, there is an option set field value available. With help of advanced find, I have generated the fetch XML query based on my requirement. 

Note: you have to use an 'alias' to retrieve the label name as shown below.

To get option set value with alias:

OptionSetVal= ((OptionSetValue)((AliasedValue)enitycollection.Entities[0].Attributes["a_optionsetaliasname"]).Value).Value;

-----------------------------------------------------------------------------------------------------

var fetchXml = @"<fetch version='1.0' output-format='xml-platform' mapping='logical' distinct='false'>
                                  <entity name='new_country'>
                                    <attribute name='new_countryid' />
                                    <attribute name='new_cityid' />
                                    <order attribute='new_cityid' descending='false' />
                                    <filter type='and'>
                                      <condition attribute='new_countryid' operator='eq' value='{0}' />
                                    </filter>
                                    <link-entity name='new_city' from='new_cityid' to='new_cityid' visible='false' link-type='inner'>
                                      <attribute name='new_cityoptionset' alias='a_cityoptionset'/>
                                    </link-entity>
                                  </entity>
                                </fetch>";
                fetchXml = string.Format(fetchXml, CountryId);//countryid index value dynamically
                Result = service.RetrieveMultiple(new FetchExpression(fetchXml));
                if (Result != null && Result.Entities.Count > 0)
                {
                    if (Result.Entities[0].Contains("a_citycode"))
                    {
                        var OptionSetValLabel = stationResult.Entities[0].FormattedValues["a_citycode"];
                    }

                }


Step 1:

 var OptionSetValLabel = stationResult.Entities[0].FormattedValues["a_citycode"];

Step 2:

public static string GetOptionSetValueLabel(string entityName, string fieldName, int optionSetValue, IOrganizationService service)
        {
 
            var attReq = new RetrieveAttributeRequest();
            attReq.EntityLogicalName = entityName;
            attReq.LogicalName = fieldName;
            attReq.RetrieveAsIfPublished = true;
 
            var attResponse = (RetrieveAttributeResponse)service.Execute(attReq);
            var attMetadata = (EnumAttributeMetadata)attResponse.AttributeMetadata;
 
            return attMetadata.OptionSetVal.Options.Where(x => x.Value == optionSetValue).FirstOrDefault().Label.UserLocalizedLabel.Label;
 

        }


Click on the link to learn more about the plugin.


That's all...

Thank you...

Keep Learning.

Tuesday, July 6, 2021

Retrieve link entity attribute value from fetch XML in dynamics 365 using Plugin C# Code

The following example code shows how to get related entities Country and City, City Code using fetch XML link-entity

Note: Here you need to specify the ‘alias’ name which you can keep different than ‘attributename’ and use the same alias name in GetAttributeValue while fetching value. let's go into detail by looking into the below example.

var citycode = countryResult.Entities[0].GetAttributeValue<AliasedValue>("a_citycode").Value.ToString();

Follow the below example to understand more on how to retrieve the link entity attribute value. In this example I am using fetch XML to get the record based on my requirement you can use advanced find to retrieve the fetch XML query as per your requirement. 

var fetchXml = @"<fetch version='1.0' output-format='xml-platform' mapping='logical' distinct='false'>
                                  <entity name='new_country'>
                                    <attribute name='new_countryid' />
                                    <attribute name='new_cityid' />
                                    <order attribute='new_cityid' descending='false' />
                                    <filter type='and'>
                                      <condition attribute='new_countryid' operator='eq' uitype='new_country' value='{0}' />
                                    </filter>
                                    <link-entity name='new_city' from='new_cityid' to='new_cityid' visible='false' link-type='outer' alias='a_8e4b53ab387aea11a811000d3a8b2b00'>
                                      <attribute name='new_citycode' alias='a_citycode'/>
                                    </link-entity>
                                  </entity>
                                </fetch>";
                fetchXml = string.Format(fetchXml, CountryId);// pass dynamics country id in the first index {0}
                var countryResult = service.RetrieveMultiple(new FetchExpression(fetchXml));
               
                if (countryResult != null && countryResult.Entities.Count > 0)
                {
                    var citycode = countryResult.Entities[0].GetAttributeValue<AliasedValue>("timezonediff").Value.ToString();
                }


Click here to learn more about Plugin dynamics 365

That's all..

Thank you .. keep learning..

Write, Register and Debug a plugin step by step in Dynamics 365

 📝Writing, registering, and debugging a plug-in in Dynamics 365 

In this blog, I will be showing How to Write, Register, and Debug a Plug-in in dynamics 365? 
Follow the complete step by step process written below:

Short Overview:
  1. Create a .NET Framework Class library project in Visual Studio
  2. Add the Microsoft.CrmSdk.CoreAssemblies NuGet package to the project
  3. Implement the IPlugin interface on classes that will be registered as steps.
  4. Add your code to the Execute method required by the interface
    1. Get references to services you need
    2. Add your business logic
  5. Sign & build the assembly
  6. Test the assembly
    1. Register the assembly in a test environment
    2. Add your registered assembly and steps to an unmanaged solution
    3. Test the behavior of the assembly
    4. Verify expected trace logs are written
    5. Debug the assembly as needed
Note: If you want to learn in a deep about plugin development click here.
            

Begin:


Writing a Plug-in