Step by Step Salesforce Tutorial – Creating Trigger and Test cases – 6 of 6

Get articles everyday as a email directly to your inbox:
Click to publicize, if you like this article :

This is the last tutorial in series and we will see that how to create a Trigger and Test Cases in salesforce.

A trigger is an Apex script that executes before or after specific data manipulation language (DML) events occur, such as before object records are inserted into the database, or after records have been deleted.

Triggers are stored as metadata in Salesforce.com. A list of all triggers in your organization is located at SetupDevelopApex Triggers. In addition to this list, triggers are associated and stored with specific objects.

To define a trigger:

  1. For a standard object, click SetupCustomize, click the name of the object, then click Triggers.For a custom object, click SetupCreateObjects and click the name of the object.For campaign members, click SetupCustomizeCampaignsCampaign MemberTriggers.For case comments, click SetupCasesCase CommentsTriggers.

    For email messages, click SetupCasesEmail MessagesTriggers.

  2. In the Triggers related list, click New.
  3. Click Version Settings to specify the version of Apex and the API used with this trigger. If your organization has installed managed packages from the AppExchange, you can also specify which version of each managed package to use with this trigger. Generally, you should use the default values for all versions. This associates the trigger with the most recent version of Apex and the API, as well as each managed package. You can specify an older version of a managed package if you want to access components or functionality that differs from the most recent package version. You can specify an older version of Apex and the API to maintain specific behavior.
  4. Select the Is Active checkbox if the trigger should be compiled and enabled. Leave this checkbox deselected if you only want to store the script in your organization’s metadata. This checkbox is selected by default.
  5. In the Body text box, enter the Apex for the trigger. A single trigger can be up to “1 million” characters in length.

Read this URL for Governor limit reference.

To define a trigger, use the following syntax:

trigger triggerName on ObjectName (trigger_events) {
   code_block
}

where trigger_events can be a comma-separated list of one or more of the following events:

  • before insert
  • before update
  • before delete
  • after insert
  • after update
  • after delete
  • after undelete

So, lets start with creating trigger.

I want that duplicate student should not be created on the basis of Email id. we can achieve this by other way also, like during creation of email field, we can specify it as unique field.

Open eclipse and right click on salesforce project and select Create new Trigger, as shown in below image

Creating Trigger in Salesforce using force.com IDE

Creating Trigger in Salesforce using force.com IDE

As you can see, to create trigger we have to select the Apex version and operations in wizard.

During Trigger creation, keep in mind that it may be required in bulk operations so governor limit may be problem.

So instead of getting query for all triggers individually, I created an array and after that created a set of email entered for all records, and loop through all records invidually and checked for duplicity.
Only one SOQL is fired instead of all triggers individually and thus can work for bulk insert.

For SOQL Best Practice refer :
http://wiki.developerforce.com/index.php/Best_Practice:_Avoid_SOQL_Queries_ Inside_FOR_Loops

trigger DuplicateStudentCheck on Student__c (before insert) {

       //Get all Student__c related to the incoming Student records in a single SOQL query.
       Student__c[] studentsList = Trigger.new;
       Set emailSet = new Set();
       for(Student__c s : studentsList)
       {
       	emailSet.add(s.Email__c);
       }

       //Get list of duplicate Students
       List duplicateStudentList = [Select s.Name, s.Email__c From Student__c s
where s.Email__c IN :emailSet];

       Set duplicateEmailIds = new Set();

       for(Student__c s : duplicateStudentList)
       {
       	duplicateEmailIds.add(s.Email__c);
       }

       for(Student__c s : studentsList)
       {
       	    if(duplicateEmailIds.contains(s.Email__c))
       	    {
       	    	s.Email__c.addError('Record already exist with same email Id');
       	    }
       }
}

Note: You can add, edit, or delete Apex using the Salesforce.com user interface only in a Developer Edition organization, a Salesforce.com Enterprise Edition trial organization, or sandbox organization. In a Salesforce.com production organization, you can only make changes to Apex by using the Metadata API deploy call, the Force.com IDE, or theForce.com Migration Tool. The Force.com IDE and Force.com Migration Tool are free resources provided by salesforce.com to support its users and partners, but are not considered part of our Services for purposes of the salesforce.com Master Subscription Agreement.

Test Cases in Salesforce :


Test case integral part of code developement.

  • You must have at least 75% of your Apex scripts covered by unit tests to deploy your scripts to production environments. In addition, all triggers should have some test coverage.
  • Salesforce.com recommends that you have 100% of your scripts covered by unit tests, where possible.
  • Calls to
    System.debug 

    are not counted as part of Apex code coverage in unit tests.

So, here we are going to create Test Case for trigger which we have written:

@isTest
private class TestTriggers {

    static testMethod void myUnitTest() {
        Student__c s = new Student__c();
        s.Name = 'Om Test';
        s.l_Name__c = 'LastName';

        Course__c c = new Course__c();
        c.Name = 'SFDC';
        c.Fees__c = 2000;
        insert c;

        s.Course__c = c.Id;
        s.Installment_1__c = 2000;
        s.Email__c = 'admin@shivasoft.in';
        try
        {
        	insert s;
        }
        catch(System.DMLException e)
        {
        	System.assert(e.getMessage().contains('Record already exist with same email Id'));
        }
    }
}

To run the test case from Eclipse, right click on test class as shown in below image:

Run Test Case using Eclipse in Salesforce

Run Test Case using Eclipse in Salesforce

The output of the test case, Test coverage result:

Test Coverage result Eclipse in Salesforce

Test Coverage result Eclipse in Salesforce

Run test cases from salesforce.com browser:

click SetupDevelopApex Classes, click the name of the class, then click Run Test. If your class calls another class or causes a trigger to execute, those Apex scripts are included in the total amount used for calculating the percentage of code covered.

To run all the unit tests in your organization, click SetupDevelopApex Classes, then click Run All Tests.

You can leave a response, or trackback from your own site.
  • sa

    Can u Explain test method and annotations in Apex in salesforce

  • kiran

    what is the main difference between before trigger and after trigger

  • Shashy kala

    Hi please someone help me .

    How to write Test Class for this Trigger

    trigger BillingAddPopulateOnCustomAddress on Account (before insert, before update) {

    for(Account acct : Trigger.new){

    if((acct.cust_Address_1__c==null && acct.cust_City__c==null &&
    acct.cust_State__c==null && acct.cust_Country__c==null &&
    acct.cust_Postal_Code__c==null) &&

    ( acct.BillingStreet !=null || acct.BillingCity !=null ||
    acct.BillingCountry !=null || acct.BillingPostalCode !=null ||
    acct.BillingCountry !=null )) {

    //Moving Billing Address to CustomAddress
    acct.cust_Address_1__c = acct.BillingStreet;
    acct.cust_City__c = acct.BillingCity;
    acct.cust_State__c = acct.BillingState;
    acct.cust_Country__c = acct.BillingCountry;
    acct.cust_Postal_Code__c = acct.BillingPostalCode;

    }
    }
    }

  • ramesh

    what is difference b/w sales cloud and service cloud.

    what type of cloud salesforce.com?

  • http://www.facebook.com/profile.php?id=1627666048 Aruna Vemula

    Can I write this test class in Salesforce.com….If yes where can I write ode for a particular apex class or a trigger

    • JitendraZaa

      Hi Aruna,
      You can write test class in Developer, Enterprise and Unlimited Edition.

      Go to SetuP | Develope | Apex class

  • Amitendra Dixit

    HI
    I have found error when I am parsing a wsdl into salesforce, the error is
    Unable to find schema for element; {http://www.w3.org/2001/XMLSchema}boolean

    Please solve it out why this is happening.

    thanks
    Amitendra

  • Amitendra Dixit5

    Hi
    I have declear a list as below , but getting error when I am using it
    public List serviceObjects = new List(); 
    Error: Compile Error: Variable does not exist: serviceObject at line 22 column 25

    please let me know why its happen

  • smita

    Hi Jeetendra,
    This is smita .New to salesforce software in my org.
    Wanted to check 2 things ,looking for some help on this regards:
    I have created around 70 leads with the for our company’s client.
    Now if i want can i set-up a remainder or alert wherein i get a alert or remainder popup to update the status with the latest status,or for that matter even if i have to follow-up remainder or alert mechanism.

    • JitendraZaa

      Hi Smita,
      There are lots of ways but i am explaining the easiest way in which no Apex involved.
      Steps:
      1: Create temp field of checkbox type. Dont add this field on any page layout, it should be hidden.
      2.Create workflow rule to update (toggle) that field.
      3.Create other time dependent workflow rule which will check last modify date with current date. if record not modified in last 10 days then send email.

      Note : For existing leads, you will need to update ll the leads using apex.

  • Amitendra Dixit

    Hi

    I am to create a report that I have attached in the image ..
    please help me, It is cutom type report. 

    Thanks,

    • JitendraZaa

      Hi dont think that it is possible using Report builder. Either your schema should support this or You need to write custom VF Page.

  • Amitendra Dixit

    Hi
    I am to move the data of a encrypted field from on object to other object. But encrypted field has many restriction like, we can not use it in farmula field. So please tell me what is the option for this.
    please let me no if any option for this.

    Thanks,

    • JitendraZaa

      Hi Amitendra,
      You can use Trigger for New records. And if you want to create Objects from existing then you can use Developer console to run Anonymous APEX code

      • Amitendra Dixit

         Hi Jitendra,

        Thanks for your suggetion. This possible by trigger. But there is no way exxcepting
        trigger. Like, can we do it by using VF page or anyother method?
        If any option please tell me …

        Thanks,

        • JitendraZaa

          Then you can use Schedular Class.

  • Amitendra Dixit

     Hi Jitendra,

    I am to use a tool “DBAmp” in my application. This is a paid tool. It can intigrate with Sql Server 2005 version or later. We can get backup of salesforce app data in sql using this tool.

    I have problem in configuration and login by this tool. will you help me about this..?, Please..

  • Amitendra Dixit

    Hi I have installed package RelationShip Group( households) … There is related list on the detailed page of this package object.. The list name is Review Schedule, but this object have no relationship with the object Review_Schedule or the objects that are child of Review Schedule…

    So in the Related List how the record is visible… Please help me …
    tell how its happen… Thanks..

    • JitendraZaa

      Hi Amitendra,
      There must be lookup field in Child Object for Parent.
      If its not there then it possible that they are using custom VF page.
      But i am sure that there is lookup field in child object. You can send me the screen shot, then i can answer in better way.

      • Amitendra Dixit

         Thanks Jitendra for your information.

        I have solved this problem. Actually there was a object which had lookup with RelationShip Group object and master Details with other object. Therefore it is happen, without direct relationship. Relations between object like this image.

  • Sakshigandhi261

    hii,
     plzz tel me another way to write error message in triger except this:
    trigger test on Account (before insert) {
    trigger.new[0].name.addError(‘Nice try’);
    }

    • JitendraZaa

      Hi,
      Either you can add error message at field level or object level.

      Your code explains the field level, below example is at object level:

      trigger test on Account (before insert) {trigger.new[0].addError(‘Nice try’);} 

      • Thippirisettyganesh

        hi jitendra, i posted the question in boards.com please see that one. and give the reply

  • Ravikumaru007

     how to generate Mr. before all records using apex trigger.

    plz… send me code for my mail     ravikumaru007@gmail.com

  • Amarnagumanni

    which language is the relation of salesforce

  • Lakshmi Sf6

    sorry the other one is luxmi085@gmail.com

  • Lakshmi Sf6

    hi 
    i am Lakshmi
    I started learning salesforce
    i completed ECE ,bcz of electronics dept im not able to understand your code
    i am poor in programming,
    i want to learn apex language
    could u pls provide me some material
    in sales force for developer what should i need to learn
    i am unable to understand triggers code
    provide me some simple examples  or material to understand.
    i am afraid of coding and programming, but i wish to learn it and to get ajob in salesforce as a developer,
    my mail id : lakshmi.sf6@gmail.com 
                      luxmi85@gmail:disqus.com

    Thanks & Regards
    lakshmi.

    • Surendrak188

       there are many videos uploaded in you tube as a beginner to the sales force you can get some basic  idea later u can go fr documentation hopes it helps u

      Regards,
      Surendra

  • Arun

    07204651195

  • http://twitter.com/mjgallag24 Michael Gallagher

    @58580936bae722b65524173157e34bc3:disqus in looking at your code, I’m fairly certain that if I insert two students with the same email address, in the same batch, it will allow them both to be inserted

    • Anonymous

      Hey Michael,
      Thanks for reviewing the code.
      I have tested code and its working fine. Any reason or bug you find in above code?

      • http://twitter.com/mjgallag24 Michael Gallagher

        Hi Jitendra,
        I’m referring to batch DML, for example something like:insert new Student__c[]{new Student__c(Email__c=’notInDatabase@ notInDatabase.com’), new Student__c(Email__c=’notInDatabase@ notInDatabase.com’)};
        Both student records will be inserted into the database at the same time, as part of the same batch, with duplicate email addresses. See what I mean now?
        Regards,
        Mike

        • Anonymous

          Yeah.. You are right. I am comparing with existing records only and skipped the condition to check duplicate email Id within batch.

          Thank you so much for your time.

          Regards,
          Jitendra Zaa

  • Ashish yadav

    A single trigger can be up to 32,000 characters in length is wrong its 1 million not 32000 please correct

    • http://shivasoft.in Jitendra Zaa

      Thanks Ashish for pointing it out. It was 32,000 at the time when i wrote this article. Now its updated.

      Regards,
      Jitendra Zaa

  • Himanshu

    nope

  • Himanshu

    It works fine in apex page but on native page it doesn’t come as html

  • HImanshu

    I am facing one issue. I want to show trigger error message as html for example. I have following code

    c.addError(‘cannot be rejected through this screen.click < a href="#" rel="nofollow" >here< /a > to Approve/Reject’);

    But it shows as normal text without any hyperlink.

    • http://shivasoft.in Jitendra Zaa

      Hi Himanshu,
      It works perfectly. Check below code snap:

      trigger AddLinkInError on Student__c (before insert) {
      Student__c[] studentsList = Trigger.new;
      for(Student__c s : studentsList )
      {
      s.adderror('< a href = \' www.shivasoft.in \ ' > This is Test < / a >. ');
      }
      }

      Regards,
      Jitendra Zaa

    • http://shivasoft.in Jitendra Zaa

      Hi Himanshu,
      Any luck on this problem ?

      Regards,
      Jitendra Zaa

  • priya

    Hi Jitendra,

    I was getting the error only when i tried with ‘Date’.So when i try giving like this,

    CREATEDDATE=\’9/5/2011 12:00:00 AM\’

    i was getting the same error.

  • priya

    Hi Jitendra,
    i just want to make sure is this the format u’ve suggested while fetching from sfdc?

    CREATEDDATE=\’9/5/2011 12:00:00 AM\’

    Regards,
    Priya

    • http://shivasoft.in Jitendra Zaa

      Yes,
      Example –
      global final String gstrQuery = ‘SELECT ID, StageName, Designated_Contact_Count__C from Opportunity where StageName != \’Expired\”;

      Please reply that it worked or not?
      Regards,
      Jitendra Zaa

  • priya

    am getting this error,no viable alternative at character ” when i try fetching records from salesforce.Any help would be greatly appreciated.

    • http://shivasoft.in Jitendra Zaa

      Hi Priya,
      Instead of double quotes (“) , use single Quotes (‘)
      OR
      in SOQL query use escape sequence – instead of (‘) use (\’)

      Regards,
      Jitendra Zaa

  • omni

    Error: Compile Error: expecting a right parentheses, found ‘status__c’ at line 5 column 17

  • omni

    now this error

    Error: Compile Error: line 5:17 no viable alternative at character ‘’’ at line 5 column 17

  • omni

    this is the code

    trigger trg on position__c (after insert,after update)
    {

    position__c ps = [ select Remainings__c,Openings__c from position__c];
    if(trigger.new(0)status__c==’recruited’)
    {
    ps.Remainings__c = ps.Openings__c – 1;
    }
    update ps;

    }

    nd the error i get isError: Compile Error: expecting a right parentheses, found ‘status__c’ at line 5 column 17

    • http://shivasoft.in Jitendra Zaa

      Hi try below code :

      trigger trg on position__c (after insert,after update)
      {

      position__c ps = trigger.new[0];
      if(ps.status__c==’recruited’)
      {
      ps.Remainings__c = ps.Openings__c – 1;
      }
      update ps;

      }

  • http://shivasoft.in Jitendra Zaa

    Hi Omni,
    If Both object are in relation then u can use “before Trigger” on object (position).

    Thanks,
    Jitendra Zaa

  • omni

    ohk sir i need to update my object(position) no_of_openings__c which is a picklist whn the status STATUS=recruited of another object (job application)
    no_of_openings__c=Remainings__c-1; how do i proceed im a newbie

  • Narasimha

    Ya its good!
    Thanku so much.

    • http://shivasoft.in Jitendra Zaa

      Thanks Narasimha.

  • sreekanth

    nice info, thanks a lot its very useful..

  • http://shivasoft.in admin

    Hi Cloud-Ling,
    Please try this:

    trigger ErrorDuplicateOutput on Account (before insert, before update)
    {
    List accList = Trigger.New;

    Set b = new Set{‘Express Logistics and Transport’,’Edge Communications’};

    for (Account acc1 : accList)
    {
    if (b.contains(acc1.Name))
    {
    acc1.Name.addError(‘This is error Message’);
    }
    }
    update accList;
    }

    /*
    * You cannot add error message in existing object, you can add message on object on which operation is performed.
    */

    Thanks,
    Jitendra Zaa

  • Cloud-Ling

    Hi, thanks for your post, it’s really helpful especially with newbies like me. How come my addError method doesn’t come up with it should’ve done?
    here’s the error message every time i try my code

    Apex trigger ErrorDuplicateOutput caused an unexpected exception, contact your administrator: ErrorDuplicateOutput: execution of BeforeInsert caused by: System.FinalException: SObject row does not allow errors: Trigger.ErrorDuplicateOutput: line 11, column 5

    here’s my code where I compiled with no errors,

    trigger ErrorDuplicateOutput on Account (before insert, before update)
    {
    List accList =new List();
    Set b = new Set{‘Express Logistics and Transport’,'Edge Communications’};
    accList = [Select Id, Name FROM Account Where Name in: b];

    for (Account acc : accList)
    {
    if (b.contains(acc.Name))
    {
    acc.Name.addError(”);
    }
    }
    update accList;
    }

    your help will be very much appreciated!
    Cloud-Ling

    • Rameshreddygunda

      can any one help me in writting a trigger for related list concept….I had Accounts and for it related list i had contacts and if i fill the Accounts and click on save i get the realated list contact and if i enter the same birthday (B.date field)  for two contact records it should throw an error Make sure that you are working on the same Account related list.