Skip to content

Nebula-Consulting/nebula-object-transformer

Repository files navigation

Nebula Object Transformer

Transform your objects! Configure your transformations using custom metadata records!

A common requirement, particularly in integrations is to be able to transform instances of one type of objects into another. For example, we might want to transform a Contact record into a JSON object represented by a map. Or we might want to do the reverse. Or we might need to transform between differed types of SObject, or between JSON objects. As part of the transformation, we might need to change the field names and/or apply functions to the values.

The Nebula Object Transformer provides a generic method to do all of that, driven by custom metadata configuration.

Since it is built on top of Nebula Core, it supports deep references in the field names. e.g. a field reference on Contact could be Account.Name. As a source field, this will be read correctly for an SObject or Map, as long as the data is actually there. Similarly, this format can be used as a target in JSON objects to implicitly construct a nested object.

Installation

  • Package entry "Nebula Object Transformer": "04tQB00000001iTYAQ"
  • Installation URL /packaging/installPackage.apexp?p0=04tQB00000001iTYAQ

Examples

The following examples are based on TransformationTest. In the package tests, custom metadata records are created in memory before running the transformation. The examples below are more like real-world usage where the custom metadata is queried by name.

The examples below show the custom metadata records in tabular format. Note that the custom metadata records in the tables are not part of the package. You would create your own custom metadata for your own requirements.

Lead to Contact

Custom Metadata (Lead_to_Contact):

Source Field Target Field
FirstName FirstName
LastName LastName

Code:

nebc.Transformation thisTransformation = new nebc.Transformation('Lead_to_Contact', Contact.class);

Lead theLead = new Lead(FirstName = 'A', LastName = 'B');
Contact newContact = (Contact) thisTransformation.call(theLead);

System.assertEquals(theLead.FirstName, newContact.FirstName);
System.assertEquals(theLead.LastName, newContact.LastName);

In this example, input and output are similar (both SObjects) and they don't require any transformation on the values. Source values are copied to the target for each field. The Type passed to the Transformation constructor allows the resulting SObject to be created correctly.

Null input values are fine

As you would hope, null values on the input record are no problem. So, we can repeat the same transformation, with an empty Lead.

nebc.Transformation thisTransformation = new nebc.Transformation('Lead_to_Contact', Contact.class);

Lead theLead = new Lead();
Contact newContact = (Contact) thisTransformation.call(theLead);

System.assertEquals(null, newContact.FirstName);
System.assertEquals(null, newContact.LastName);

Same field, multiple destinations

It's also fine to send the same input field to multiple field in the output object.

Custom Metadata (Lead_to_Contact_Description):

Source Field Target Field
FirstName FirstName
LastName LastName
FirstName Description
nebc.Transformation thisTransformation = new nebc.Transformation('Lead_to_Contact_Description', Contact.class);

Lead theLead = new Lead(FirstName = 'A', LastName = 'B');
Contact newContact = (Contact) thisTransformation.call(theLead);

System.assertEquals(theLead.FirstName, newContact.FirstName);
System.assertEquals(theLead.LastName, newContact.LastName);
System.assertEquals(theLead.FirstName, newContact.Description);

Contact to JSON

When the output type should be a JSON object, this is achieved by specifying Map<String, Object>.class in the constructor of nebc.Transformation. In this example, we transform a Contact to JSON object.

Custom Metadata (Contact_to_JSON):

Source Field Target Field
FirstName first_name
LastName last_name
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_to_JSON', Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', LastName = 'b');
Map<String, Object> newMap = (Map<String, Object>) thisTransformation.call(theContact);

System.assertEquals(theContact.FirstName, newMap.get('first_name'));
System.assertEquals(theContact.LastNamne, newMap.get('last_name'));

Contact with relationship field to JSON

When the input is an SObject, and there are lookup fields, we can use deep references to read from them. Note that extra data is never queried, so you must make sure that you have all the required fields yourself (see How to query for transformation)

Custom Metadata (Contact_deep_to_JSON):

Source Field Target Field
FirstName first_name
LastName last_name
Account.Name company
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_deep_to_JSON',  Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', LastName = 'B', Account = new Account(Name = 'ACME'));
Map<String, Object> newMap = (Map<String, Object>)thisTransformation.call(theContact);

System.assertEquals(theContact.FirstName, newMap.get('first_name'));
System.assertEquals(theContact.LastName, newMap.get('last_name'));
System.assertEquals(theContact.Account.Name, newMap.get('company'));

Transformation functions

When some values need to be modified during the transformation, we can provide the name of an Apex class to perform the transformation. In this example, we are taking a Date from Contact and transforming it into a String that can be used in JSON:

Custom Metadata (Contact_to_JSON_transform):

Source Field Target Field Apex Class
FirstName first_name
LastName last_name
Birthdate date_of_birth nebc.JsonSerialize
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_to_JSON_transform', Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', LastName = 'B', Birthdate = Date.today());
Map<String, Object> newMap = (Map<String, Object>) thisTransformation.call(theContact);

System.assertEquals(theContact.FirstName, newMap.get('first_name'));
System.assertEquals(theContact.LastNamne, newMap.get('last_name'));
System.assertEquals(JSON.serialize(theContact.Birthdate), newMap.get('date_of_birth'));

The transformation function we needed already exists in Nebula Core, but you can always write your own class. It just has to implement nebc.Function and be global. It should expect to receive an Object which is the value from the input field.

Contact to JSON with deep maps

By using dots in the target fields, we can construct a map with sub-objects inside it.

Custom Metadata (Contact_to_JSON_deep):

Source Field Target Field
FirstName person.first_name
LastName person.last_name
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_to_JSON_deep', Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', LastName = 'B');
Map<String, Object> newMap = (Map<String, Object>) thisTransformation.call(theContact);

Map<String, Object> person = (Map<String, Object>) newMap.get('person');
System.assertEquals(theContact.FirstName, person.get('first_name'));
System.assertEquals(theContact.LastName, person.get('last_name'));

Deep map to Lead

Similarly, a JSON object with nested objects can be read directly as input for a transformation.

Custom Metadata (JSON_to_Lead_deep):

Source Field Target Field
person.first_name FirstName
person.last_name LastName
nebc.Transformation thisTransformation = new nebc.Transformation('JSON_to_Lead_deep', Lead.class);

Map<String, Object> theMap = new Map<String, Object>{
        'person' => new Map<String, Object>{
                'first_name' => 'A', 'last_name' => 'B'
        }
};
Lead newLead = (Lead) thisTransformation.call(theMap);

Map<String, Object> person = (Map<String, Object>) theMap.get('person');
System.assertEquals(person.get('first_name'), newLead.FirstName);
System.assertEquals(person.get('last_name'), newLead.LastName);

Constants in the metadata

By using a constant transformation function, we can set constant values in the metadata. More direct support for constants could be useful in a future version.

Custom Metadata (Lead_to_Contact_generate):

Source Field Target Field Apex Class Apex Class Parameters
FirstName FirstName
LastName LastName
Description nebc.StringConstant { "value": "a metadata constant!"}
Transformation thisTransformation = new Transformation('Lead_to_Contact_generate', Contact.class);

Lead theLead = new Lead(FirstName = 'A', LastName = 'B');
Contact newContact = (Contact)thisTransformation.call(theLead);

System.assertEquals(theLead.FirstName, newContact.FirstName);
System.assertEquals(theLead.LastName, newContact.LastName);
System.assertEquals('a metadata constant!', newContact.Description);

Whole-object transformation functions TBD

Sometimes the function to modify values during the transformation might need the whole input object to calculate an output value e.g. for concatenating two text/string fields into one, or for doing a calculation. In that case, you can set the "Apex Class Receives" Parameter in the custom metadata. In this example, we concatenate the first and last name into the description field.

Custom Metadata (Lead_to_Contact_whole_object):

Source Field Target Field Apex Class Apex Class Receives
FirstName FirstName
LastName LastName
FirstName Description FirstNameAndLastName Whole Object
nebc.Transformation thisTransformation = new nebc.Transformation('Lead_to_Contact_whole_object', Contact.class);

Lead theLead = new Lead(FirstName = 'A', LastName = 'B');
Contact newContact = (Contact)thisTransformation.call(theLead);

System.assertEquals(theLead.FirstName, newContact.FirstName);
System.assertEquals(theLead.LastName, newContact.LastName);
System.assertEquals(theLead.FirstName + ' ' + theLead.LastName, newContact.Description);

In this case, the Apex Class will receive a nebc.Tuple containing the metadata for this field, and the whole input object i.e. <nebc__Transformation_Field__mdt, Lead>. So, the implementation of FirstNameAndLastName is as follows

global class FirstNameAndLastName implements nebc.Function {

    public Object call(Object o) {
        nebc.Tuple tuple = (nebc.Tuple)o;
        SObject inputSObject = (SObject)tuple.get(1);

        return inputSObject.get('FirstName') + ' ' + inputSObject.get('LastName');
    }
}

Reverse transformations

You can easily create a reverse transformation using the same metadata as for the forwards transformation. So, we can re-use the metadata from Contact to JSON to do a round-trip

Custom Metadata (Contact_to_JSON):

Source Field Target Field
FirstName first_name
LastName last_name
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_to_JSON',  Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', Birthdate = Date.today());
Map<String, Object> newMap = (Map<String, Object>)thisTransformation.call(theContact);

System.assertEquals(theContact.FirstName, newMap.get('first_name'));
System.assertEquals(theContact.LastName, newMap.get('last_name'));

nebc.Transformation reverseTransformation = new nebc.ReverseTransformation('Contact_to_JSON', Contact.class);

Contact roundTripContact = (Contact)reverseTransformation.call(newMap);

System.assertEquals(theContact.FirstName, roundTripContact.FirstName);
System.assertEquals(theContact.LastName, roundTripContact.LastName);

This offers the same functionality as a forward transformation. It simply swaps some metadata fields around in the constructor.

Reverse transformation with transformation function

To reverse a transformation where you provided an Apex Class to transform the data, you may need to provide a Reverse Apex Class in the custom metadata. For example, we can do a round trip on Transformation functions from earlier.

Custom Metadata (Contact_to_JSON_transform_reversible):

Source Field Target Field Apex Class Reverse Apex Class Reverse Apex Class Parameters
FirstName first_name
LastName last_name
Birthdate date_of_birth nebc.JsonSerialize DeserializeToNamedType { "typeName": "Date" }
nebc.Transformation thisTransformation = new nebc.Transformation(transformationFieldMetadata,  Map<String, Object>.class);

Contact theContact = new Contact(FirstName = 'A', Birthdate = Date.today());
Map<String, Object> newMap = (Map<String, Object>)thisTransformation.call(theContact);

System.assertEquals(theContact.FirstName, newMap.get('first_name'));
System.assertEquals(JSON.serialize(theContact.Birthdate), newMap.get('date_of_birth'));

nebc.Transformation reverseTransformation = new nebc.ReverseTransformation(transformationFieldMetadata, Contact.class);

Contact roundTripContact = (Contact)reverseTransformation.call(newMap);

System.assertEquals(theContact.FirstName, roundTripContact.FirstName);
System.assertEquals(theContact.Birthdate, roundTripContact.Birthdate);

Where the actual transformation function is defined as:

public class DeserializeToNamedType implements Function {

    private String typeName; // Assigned via CMDT parameters
    private Type type {
        get {
            if(type == null) {
                type = TypeLoader.getType(typeName);
            }
            return type;
        }
        set;
    }

    public Object call(Object o) {
        return JSON.deserialize((String)o, type);
    }
}

Ignoring some fields

If you are doing a round-trip transformation, then you may include some fields that only work in one direction. You can describe this by using Target_Is_Read_Only__c and Source_Is_Read_Only__c in the custom metadata.

When the target is marked as ready-only, then the transformation will not attempt to write to it. You might still want to include for when you reverse the transformation in the second half of your round trip e.g. for trannsforming the last modified date of a record in an external system onto a custom field in Salesforce. You can read that date from the external system, but you should not write it back later.

When the source is marked as read-only, this has no effect on the straightforward transformation. However, when you do the reverse transformation, this turns into the target being read-only. In which case this field is not written to during a reverse transformation e.g. when reading from a formula field on an SObject and transforming that to send to an external system. When you reverse the transformation to back from the external system to the SObject, you must not write back to the formula field.

Note that the notion of read only is on the in-memory object. Record Id for an SObject is an example that you would not want to mark as read-only. In database terms, the Id field is not updateable. But you need that value for the in-memory object if you are going to do an update to the database.

How to query for transformation

When the source record is an SObject, it is useful to know which fields need to be queried. Once a nebc.Transformation instance is initialised, you can get the field list using getSourceFields() e.g.

Custom Metadata (Contact_deep_to_JSON):

Source Field Target Field
FirstName first_name
LastName last_name
Account.Name company
nebc.Transformation thisTransformation = new nebc.Transformation('Contact_deep_to_JSON',  Map<String, Object>.class);

Set<String> fields = thisTransformation.getSourceFields();
System.assertEquals(3, fields.size());
System.assert(fields.contains('FirstName'));
System.assert(fields.contains('LastName'));
System.assert(fields.contains('Account.Name'));

If you're familiar with Nebula Core, you can use nebc.QueryBuilder to generate a query e.g.

String query = new nebc.QueryBuilder(Contact.SObjectType)
        .addFields(fields)
        .setWhereClause('Id = :anId')
        .getQuery();

About

Transform your objects using custom metadata configuration

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages