在插入/更新Apex触发之前,测试覆盖率失败

时间:2013-10-10 13:04:04

标签: triggers salesforce code-coverage apex-code

我在Opportunity上的插入/更新触发器之前有这个非常简单,它根据包含Sales Office(State)位置信息的下拉值自动选择Price Book。

这是我的触发器

trigger SelectPriceBook on Opportunity ( before insert, before update ) {

    for( Opportunity opp : Trigger.new ) {
        // Change Price Book
        // New York
        if( opp.Campus__c == 'NYC' )
          opp.Pricebook2Id = PB_NYC; // contains a Pricebook's ID

        // Atlanta
        if( opp.Campus__c == 'ATL' )
          opp.Pricebook2Id = PB_ATL; // contains another Pricebook's ID
    }
}

这是我的测试类

@isTest (SeeAllData = true)
public class SelectPriceBookTestClass {

    static testMethod void validateSelectPriceBook() {

        // Pricebook IDs
        ID PB_NYC =  'xxxx';
        ID PB_ATL =  'xxxx';

        // New Opp
        Opportunity opp = new Opportunity();
        opp.Name = 'Test Opp';
        opp.Office__c = 'NYC';
        opp.StageName = 'Quote';       

        // Insert
        insert opp;

        // Retrive inserted opportunity
        opp = [SELECT Pricebook2id FROM Opportunity WHERE Id =:opp.Id];
        System.debug( 'Retrieved Pricebook Id: ' + opp.Pricebook2Id );

        // Change Campus
        opp.Office__c = 'ATL';
        // Update Opportunity
        update opp;

        // Retrive updated opportunity
        opp = [SELECT Pricebook2id FROM Opportunity WHERE Id =:opp.Id];
        System.debug( 'Retrieved Updated Pricebook Id: ' + opp.Pricebook2Id );        

        // Test
        System.assertEquals( PB_ATL, opp.Pricebook2Id );

    }
}

测试运行报告 0%测试覆盖率

此外,在类似的行上,我有另一个插入触发器,它将事件的所有者设置为与父Lead的所有者相同。这是代码:

trigger AutoCampusTourOwner on Event( before insert ) {

    for( Event evt : Trigger.new ) {
        // Abort if other kind of Event
        if( evt.Subject != 'Visit' )
            return;        

        // Set Owner Id
        Lead parentLead = [SELECT OwnerId FROM Lead WHERE Id = :evt.WhoId];
        evt.OwnerId = parentLead.OwnerId;

    }
}

这也导致0%的覆盖率 - 我的猜测是它与两者中的 for 循环有关。我知道我通过在for循环中调用SOQL查询来严肃地藐视DML规则,但是对于我的目的,它应该没问题,因为这些事件是手动创建的,并且一次只能创建一个 - 所以没有调控器限制的范围。批量插入。

两种情况下的代码均可100%运行。请为测试用例建议修复。

3 个答案:

答案 0 :(得分:0)

你试过trigger.old吗?我的想法是,当您将测试类中的办公室从NYC更新为ATL时,值“NYC”将处于trigger.old中,这就是您要在触发器中检查的内容。 我可能是错的,因为我也是apex的新手,但试试看,让我知道会发生什么。

答案 1 :(得分:0)

对于第一次触发,不要做任何事情而只是创造机会并像这样执行。

SelectPriceBook的测试类

@isTest
    private class TriggerTestClass {

        static testmethod void selectPriceTest(){
            Opportunity opps = new Opportunity(
                Name= 'Test Opps',
                CloseDate = System.today().addDays(30),
                StageName = 'Prospecting', 
                ForecastCategoryName = 'Pipeline',
                Office__c = 'NYC');
                insert opps;

                Opportunity opps2 = new Opportunity(
                Name= 'Test Opps 2',
                CloseDate = System.today().addDays(28),
                StageName = 'Prospecting', 
                ForecastCategoryName = 'Pipeline',
                Office__c = 'ATL');
                insert opps2;
        }

    }

它会为您提供良好的测试覆盖率,而且我不知道您在AutoCampusTourOwner中想要做什么!

答案 2 :(得分:0)

我有这些测试课

在 inflooens__Client_Email__c 上触发 ClientEmailTrigger(插入后、更新后、插入前、更新前){

ApexTriggerSettings__c setting = ApexTriggerSettings__c.getValues('Inflooens Trigger Settings');

if(setting != NULL && setting.ClientEmailTrigger__c == TRUE){

    AuditTrailController objAdt = new AuditTrailController();
    
    if(Trigger.isAfter){
        if(Trigger.isUpdate){
            System.debug('In Update Client Email Record');
            objAdt.insertAuditRecord(Trigger.newMap, 'inflooens__Client_Email__c', Trigger.new.get(0).Id, Trigger.oldMap);
        }
    
        if(Trigger.isInsert){
            objAdt.insertAuditRecord(Trigger.newMap, 'inflooens__Client_Email__c', null , null);
        }
    }
}

}