Wednesday, 28 June 2017
Sunday, 18 June 2017
Trigger Best Practice
Trigger Best Practice
Querying Salesforce in Trigger :
Make sure you have no SOQL in for loop
Wrong Approach:
// Avoid SOQL in for-loop
trigger HappyTrigger on Happiness (before insert, before update) {
for(Happy__c m : Trigger.new){
User c = [SELECT Id FROM user WHERE smile__c = m.Id];
}
}
//Avoid DML operation in for-loop
trigger HappyTrigger on Happiness (before insert, before update) {
for(Happy__c m : Trigger.new){
User c = [SELECT Id FROM user WHERE smile__c = m.Id];
update c;
}
}
Recommended Approach:
Trigger HappyTriggeronInsertUpdate on Happy__c (before insert, before update) {
Set<ID> ids = Trigger.newMap.keySet();
List<User> c = [SELECT Id FROM user WHERE id in :ids];
}
Bulkify Trigger with Set and Map:
Salesforce provides standard data-structures like Maps and Set to per-filter and pre-populate data to avoid repeated SOQL calls, this allows user to handle bulk data, often called bulkification of code. Triggers in salesforce are intentionally designed to handle bulk data, so here we simple pull Ids into a set which is used to iterate and populate a Map holding all values. The map is our data storage and used for querying Salesforce
Recommended Approach
//Bulkifying trigger using set and maps example
trigger accountOnUpdateTrigger on Account (before insert) {
//for every contact in account, lets remove duplicate
Set<Id> contactIds = trigger.newMap.keySet();
//Create a Map of Id and Contact which are in the set
Map<id, Contact> contactMap = new Map<Id, Contact> [Select id, FirstName, LastName, City From Contact where Id in :contactSet];
//get all values to update from MAP instead using SOQL
for (Account acc : Trigger.new)
acc.favorite_city__c = contactMap.get(acc.contactId).city__c;
}
SOQL In for loop for large datasets :
SOQL-IN for loops, allow you to set the range of for loops through a SOQL query, example below shows two ways of writing for-loops. First approach may throw error of 'heap size' governor limit.
//SOQL a list and looping through list
//runtime exception is thrown if this query returns enough records to exceed your heap
List<Reports__C> reportLst = [Select id, report_name, created_date__c from Report__c];
for(Report__c r : reportlst)
{
//do something
}
//Effiecient way could be, querying in for loop which returns batch of 200 records at at time to process
for(Report__c r : [Select id, report_name, created_date__c from Report__c])
{
//do something
}
Avoiding SOQL Injection :
SOQL injection, allows user to pass set of keywords input, which becomes wildcard to access all data in Salesforce, if query is pro-actively designed to handle those wild card. A simple example below demonstrates SOQL injection in action.
Vulnerable to SOQL Injection:
// Query string
SELECT name FROM Account WHERE ( Name like '%Acme%');
//User Supply
test%') OR (Name LIKE '
//string becomes
SELECT name FROM Account WHERE Name LIKE '%test%') OR (Name LIKE '%')
//Result : ---> list all accounts
Dynamic SOQL:
//SOQL Injection protected
String query = '%' + name + '%';
queryResult = [SELECT Name FROM Contact WHERE (IsDeleted = false and Name like :query)];
Querying Child Items :
Inefficient way of querying child item would iterating through each parent object and querying child item, which is ugly. Smartly this can be done as show below.
Wrong Approach:
//Wrong Practices of querying child items
trigger AccountonUpdate (before update)
{
//SOQL each time in loop
for(Account ac : Trigger.New)
{
//SOQL in for loop
List<child___c> childList = [SELECT Id,Units_Sold__c,Merchandise__c
FROM child__c
WHERE Ace_account_Id = :inv.Id];
for(child__c c : liList)
{
//do something
}
}
}
Recommended Approach:
//Recommended Practices for Quering child Iteam
trigger AccountonUpdate on Account (before update)
{
//Query all child iteam outside loop
List<Account> accountItems =
[SELECT Id,Description__c,(SELECT Id,Units_Sold__c,Merchandise__c from Child__r)
FROM Account WHERE Id IN :Trigger.newMap.KeySet()];
//Loop for each account
for(Account acc : accountsItems) {
//loop for each child in account
for(child)__c c : acc.child__r) {
// Do something
}
}
@future method in Salesforce : Making integration/service communication future proof.
Annotate a method with future, when you want to perform a job asynchronously, like making an outbound call to Facebook, Twitter on a field update.Ideally if it is not 'asynchronous', then salesforce hang up and wait for response and this delay or hang up process. But key is, writing logic of @future method in separate class.
Recommended Approach : Write your @future method and logic in separate class
global class asyncApex {
@future
public static void processAccount(Set<Id> accountIds) {
List<Contact> contacts = [select id, salutation, firstname, lastname, email from Contact where accountId IN :accountIds];
for(Contact c: contacts){
System.debug('Contact Id[' + c.Id + '], FirstName[' + c.firstname + '], LastName[' + c.lastname +']');
c.Description=c.salutation + ' ' + c.firstName + ' ' + c.lastname;
}
update contacts;
}
}
Call Asynchronous class through Apex Trigger
trigger accountAsyncTrigger on Account (after insert, after update) {
//By passing the @future method a set of Ids, it only needs to be
//invoked once to handle all of the data.
asyncApex.processAccount(Trigger.newMap.keySet());
}
Things to remember
§ Future cannot call another future method neither a batch job can call @future method
§ Use callout=true for making callouts
§ You cannot guarantee order of execution
§ Future call have limits and can run concurrently
Monday, 12 June 2017
Class to change the visibility of the "chatter posts" for the "community"
@TestVisible
private void feedItemChangeAccessLevel4Communities(ClsWrappers.TriggerContext trgCtx){
if (trgCtx.isAfter){
if(!System.label.SYS_RunCaseTrigger.equalsIgnoreCase('NO')){
Map<Id,Case> caseMap = new Map<Id,Case>();
for(sobject so : trgCtx.newList){
Case c = (Case)so;
caseMap.put(c.Id, c);
}
List <feedItem> feed4CaseMap = new List<feedItem>();
Set <ID> caseIDSet = caseMap.keySet();
feed4CaseMap =[Select id, visibility, ParentId, Type from feedItem where ParentId IN: caseIDSet limit 50000];
System.debug('Test'+feed4CaseMap);
for(FeedItem item : feed4CaseMap){
String parent = item.ParentId;
if(item.Type == System.Label.FeedItem_Type && item.visibility ==System.Label.FeedItem_Visibility){
item.visibility = System.Label.FeedItem_Visibility_Internal_Users;
}
System.debug('Test'+item);
}
database.update(feed4CaseMap, false);
}
}
}
private void feedItemChangeAccessLevel4Communities(ClsWrappers.TriggerContext trgCtx){
if (trgCtx.isAfter){
if(!System.label.SYS_RunCaseTrigger.equalsIgnoreCase('NO')){
Map<Id,Case> caseMap = new Map<Id,Case>();
for(sobject so : trgCtx.newList){
Case c = (Case)so;
caseMap.put(c.Id, c);
}
List <feedItem> feed4CaseMap = new List<feedItem>();
Set <ID> caseIDSet = caseMap.keySet();
feed4CaseMap =[Select id, visibility, ParentId, Type from feedItem where ParentId IN: caseIDSet limit 50000];
System.debug('Test'+feed4CaseMap);
for(FeedItem item : feed4CaseMap){
String parent = item.ParentId;
if(item.Type == System.Label.FeedItem_Type && item.visibility ==System.Label.FeedItem_Visibility){
item.visibility = System.Label.FeedItem_Visibility_Internal_Users;
}
System.debug('Test'+item);
}
database.update(feed4CaseMap, false);
}
}
}
Query Email Template
SELECT DeveloperName,Name, Folder.Name, IsActive, LastModifiedDate FROM EmailTemplate
Thursday, 8 June 2017
Monday, 5 June 2017
Test Class for Live Chat Visitor, Live Chat Transcript and Live Chat Transcript Event.
Trigger
trigger LiveChatTranscriptEventTrigger on LiveChatTranscriptEvent (before delete) {
/**
* trigger runs
*
*/
ClsTriggerFactory.createHandler(LiveChatTranscriptEvent.sObjectType);
}
Apex Class
@name ClsTriggerLiveChatTranscriptEventHandler
public without sharing class ClsTriggerLiveChatTranscriptEventHandler extends ClsTriggerVirtual {
/**
* @see IntHelper.beforeTrigger
*/
public override void beforeTrigger(ClsWrappers.TriggerContext trgCtx) {
if (trgCtx.IsInsert){
}
if(trgCtx.isUpdate){
}
if(trgCtx.isDelete){
blockDelete(trgCtx);
}
}
/**
* @see IntHelper.afterTrigger
*/
public override void afterTrigger(ClsWrappers.TriggerContext trgCtx) {
}
/**
* Block Usere to Delete LiveChatTranscriptEvent record if his profile not in Validation_Override__c.Exclude__c
*/
@TestVisible
private void blockDelete(ClsWrappers.TriggerContext trgCtx) {
Validation_Override__c mc = Validation_Override__c.getInstance(UserInfo.getProfileId());
system.debug('mc= '+mc.Exclude__c);
if (!mc.Exclude__c){
for(sobject so : trgCtx.oldList){
LiveChatTranscriptEvent liveEvent = (LiveChatTranscriptEvent)so;
liveEvent.addError('Live Chat Transcript Event can not be deleted');
}
}
}
}
Test Class
@isTest(seeAllData = false)
public class ClsTriLiveChatTransEvtHandlr_Test{
static testMethod void getevent(){
/*
insert Live Chat Visitor
*/
LiveChatVisitor lcv = new LiveChatVisitor();
try{
insert lcv;
}catch(Exception e){}
/*
insert Live Chat Transcript
*/
LiveChatTranscript lct = new LiveChatTranscript();
lct.LiveChatVisitorid = lcv.id;
try{
insert lct;
}catch(Exception e){}
system.debug('lct' + lct);
/*
insert Live Chat Transcript Event
*/
LiveChatTranscriptEvent le = new LiveChatTranscriptEvent();
le.LiveChatTranscriptId = lct.id;
le.type = 'ChatRequest';
le.time = system.now();
try{
insert le;
}
catch(Exception e){}
try{
delete le;
}
catch(Exception e){}
}
}
trigger LiveChatTranscriptEventTrigger on LiveChatTranscriptEvent (before delete) {
/**
* trigger runs
*
*/
ClsTriggerFactory.createHandler(LiveChatTranscriptEvent.sObjectType);
}
Apex Class
@name ClsTriggerLiveChatTranscriptEventHandler
public without sharing class ClsTriggerLiveChatTranscriptEventHandler extends ClsTriggerVirtual {
/**
* @see IntHelper.beforeTrigger
*/
public override void beforeTrigger(ClsWrappers.TriggerContext trgCtx) {
if (trgCtx.IsInsert){
}
if(trgCtx.isUpdate){
}
if(trgCtx.isDelete){
blockDelete(trgCtx);
}
}
/**
* @see IntHelper.afterTrigger
*/
public override void afterTrigger(ClsWrappers.TriggerContext trgCtx) {
}
/**
* Block Usere to Delete LiveChatTranscriptEvent record if his profile not in Validation_Override__c.Exclude__c
*/
@TestVisible
private void blockDelete(ClsWrappers.TriggerContext trgCtx) {
Validation_Override__c mc = Validation_Override__c.getInstance(UserInfo.getProfileId());
system.debug('mc= '+mc.Exclude__c);
if (!mc.Exclude__c){
for(sobject so : trgCtx.oldList){
LiveChatTranscriptEvent liveEvent = (LiveChatTranscriptEvent)so;
liveEvent.addError('Live Chat Transcript Event can not be deleted');
}
}
}
}
Test Class
@isTest(seeAllData = false)
public class ClsTriLiveChatTransEvtHandlr_Test{
static testMethod void getevent(){
/*
insert Live Chat Visitor
*/
LiveChatVisitor lcv = new LiveChatVisitor();
try{
insert lcv;
}catch(Exception e){}
/*
insert Live Chat Transcript
*/
LiveChatTranscript lct = new LiveChatTranscript();
lct.LiveChatVisitorid = lcv.id;
try{
insert lct;
}catch(Exception e){}
system.debug('lct' + lct);
/*
insert Live Chat Transcript Event
*/
LiveChatTranscriptEvent le = new LiveChatTranscriptEvent();
le.LiveChatTranscriptId = lct.id;
le.type = 'ChatRequest';
le.time = system.now();
try{
insert le;
}
catch(Exception e){}
try{
delete le;
}
catch(Exception e){}
}
}
Subscribe to:
Posts (Atom)
Batch Apex
1. What are transaction limits in apex? Total number of SOQL queries issued1 - 100 Total number of records retrieved by SOQL queries - 50...
-
Trigger trigger LiveChatTranscriptEventTrigger on LiveChatTranscriptEvent (before delete) { /** * trigger runs * ...
-
1. What are transaction limits in apex? Total number of SOQL queries issued1 - 100 Total number of records retrieved by SOQL queries - 50...
-
Links https://www.shellblack.com/administration/owds/ https://www.simplilearn.com/top-salesforce-interview-questions-and-answers-artic...
If this is in regards to collections, you might have to check both:
if ( myList != null && !myList.isEmpty() ) ...
If you know that your collection is not null (because you allocated it, or it's the result of a query), then I prefer !.isEmpty() over .size() > 0:
This:
if ( !myList.isEmpty() ) ...
Not this:
if ( myList.size() > 0 ) ...
They have the same effect, but I think that .isEmpty() expresses the intent better, i.e. I want to know if the list is empty or not. I don't actually care what its size is.