Skip to content

Latest commit

 

History

History
134 lines (99 loc) · 7.48 KB

README.md

File metadata and controls

134 lines (99 loc) · 7.48 KB

Salesforce Formula Field Dependency Analysis

This package allows you to

  1. select (up to 10) Objects,
  2. view all the formula fields for those Objects, and
  3. for each formula field, view the formula and all the dependent fields.

In addition, I wanted to use this project to

  1. experiment with Dependency Injection, and
  2. start working with Lightning Web Components (LWC)

Requirements

  1. The User can select a list of Objects by entering part of the Object API name.
  2. For each matching Object that has at least one formula field, display the list of formula fields.
  3. The User can select one of the formula fields and view
    • the CustomField API name
    • the formula
    • the list of fields that the formula is dependent on
  4. If the formula field is on Object1__c and refers to the field Object2__r.Object3__r.Field__c, we must show not only the final field dependency, Object3__c.Field__c, but also all intermediate dependencies: Object1__c.Object2__c and Object2__c.Object3__c.

Installation

  1. install the unlocked package: https://login.salesforce.com/packaging/installPackage.apexp?p0=04t0a000001ahWDAAY
  2. assign the Formula_Field_Parser_Demo permission set
  3. launch the Formula Field Dependency app
  4. explore your formula fields!

Architecture - 1 - IFieldMetadataService: Getting Object and CustomField Metadata

A class with the IFieldMetadataService interface is responsible for providing

  1. a list of Object API names that match a given search key, and
  2. a map of CustomField API names to a CustomFieldWrapper
public interface IFieldMetadataService {
    List<String> getObjects(String searchKey);
    Map<String, CustomFieldWrapper> getAllFields(string objName);
}

As the name suggests, the CustomFieldWrapper wraps information about each CustomField so that the other classes don't need to know the details of how Salesforce stores all this information. This is important for Dependency Injection of the interface; you can't create an instance of Schema.SObjectField, for example, so the interface shouldn't be dependent on it.

Architecture - 2 - FormulaFieldService: Getting Information Specific to Formula Fields

The FormulaFieldService exposes one @AuraEnabled method to the LWC UI: the getFormulaFields() method searches Objects (based on the Object API name searchKey) and returns a map of Object API names to a list of formula fields for that Object. We only create a map entry if there are formula fields on the Object.

@AuraEnabled(cacheable=true)
public static Map<String, List<CustomFieldWrapper>> getFormulaFields(String searchKey)

The service also exposes three methods for querying information about a specific field:

public static Boolean isValidField(String objName, String fieldName) 
public static Boolean isReferenceField(String objName, String fieldName) 
public static String getReferenceTarget(String objName, String referenceFieldName) 

Architecture - 3 - FormulaFieldParser: Doing the Work

The FormulaFieldParser exposes one @AuraEnabled method to the LWC UI: the getFormulaDependencies() method takes a formula string and the base object for the formula field and returns a list of CustomField API names that the formula refers to.

@AuraEnabled(cacheable=true)
public static List<String> getFormulaDependencies(String objName, String calculatedFormula) {

The parser loops through the string using this Regex matching pattern: ([a-zA-Z0-9_\\.]+).

  • The pattern will pick up any CustomField API names like Object1__r.Object2__r.Field__c.
  • The pattern will also pick up any formula functions like 'TEXT' and 'ISPICKVAL'. That's why we need the isValidField method (above); so that we can throw away these keywords as we traverse the formula.

Each matched pattern (eg, Object1__r.Object2__r.Field__c) we call a 'word'. We split each word on '.' to create a list of tokens that represent the word. In our example, the list of tokens would be {'Object1__r','Object2__r','Field__c'}.

We recursively traverse the list of tokens:

  • If there is only one token, then it's either a field or a formula keyword, and we're done.
  • If there is more than one token, then the first token is a reference to another object. We get the object using the getReferenceTarget() method (above) and continue traversing the list.

Architecture - 4 - Event Model

Object Search

We reused the logic in the LWC Recipes for the Object search. The lwcObjectSearch component is wired to pass the onchange event handler's search key to the FormulaFieldService.getFormulaFields() method. Recall that this method returns a map of objects to formula fields. This map is rendered as an LWC Tree.

Selecting a Formula Field

When a formula field is selected on the Tree, the lwcObjectSearch component's onselect event handler dispatches a custom formulafieldselected event with the field (CustomFieldWrapper) as payload.

The formulaFieldDisplay component listens for the formulafieldselected event. The component is wired to the FormulaFieldParser.getFormulaDependencies() method and displays the API name, formula, and dependencies for the selected field.

Architecture - 5 - Dependency Injection

Unit testing is tricky. We need dependable, reproducible formula fields. And we obviously don't want to package custom objects and fields just for testing. So we need to mock the FieldMetadataService and use the mock service for testing.

Using Custom Metadata for Dependency Injection

The force-di package makes extensive use of Custom Metadata to implement Dependency Injection. For this exercise, I implemented a very simple pattern:

  • Class_Injection__mdt holds the record that binds a concrete class to an interface, realizing the dependency injection
  • the Label field holds the name of the interface
  • the Concrete_Class__c field holds the name of the concrete class to be injected

In the FormulaFieldService class, this is implemented as:

private static final String FIELD_METADATA_SERVICE_INTERFACE = 'IFieldMetadataService';
private static IFieldMetadataService fieldMetadataService;

static {
    List<Class_Injection__mdt> injection = [SELECT Concrete_Class__c 
                                            FROM Class_Injection__mdt 
                                            WHERE Label = :FIELD_METADATA_SERVICE_INTERFACE];
    String concreteClass = injection[0].Concrete_Class__c;                                        
    fieldMetadataService = (IFieldMetadataService) Type.forName(concreteClass).newInstance();
}

Using a Setter to Inject Dependency

The Custom Metadata binding works great. But not for testing, because you can't do DML on metadata records in a unit test. (I should examine the force-di package to see how testing is intended to be done.)

For this exercise, I just created a private @TestVisible method to inject the dependency. In the FormulaFieldService class:

// use this to inject the mock metadata service for testing
@TestVisible
private static void fieldMetadataServiceOverride(IFieldMetadataService service) {
    fieldMetadataService = service;
}

Note: normally this would be called setFieldMetadataService, but I wanted to emphasize that by using this, you're overriding whatever dependency has been injected via the Class_Injection__mdt record.

And in the test classes:

FormulaFieldService.fieldMetadataServiceOverride(new FieldMetadataServiceMock());

Class Diagram

class diagram