Apex Test Class - How to set a rollup count field in a test class - salesforce

Trying to set a value for a roll up summary field in the test class to improve code coverage. How do I do it?
public class clsPreferredIASetExt {
List<PreferredIA__c> preferredias;
public static PreferredIA__c[] tobeClosed = new PreferredIA__c[0];
public static PreferredIA__c[] newPreIAs = new PreferredIA__c[0];
public static PreferredIA__c loopadd;
public static PreferredContact__c[] contactlists = new PreferredContact__c[0];
public static Account[] InvoicedAccounts = new Account[0];
public static PreferredIA__c[] monkey;
public clspreferrediaSetExt(ApexPages.StandardSetController controller) {
preferredias = (List<PreferredIA__c>) controller.getSelected();
}
public void getInitCloseInv() {
tobeclosed = [select id, Account__c, Account__r.id, Account__r.Name,
Account__r.AccountNumber, Specialist__c,
PreferredInvoice__c, Status__c
from PreferredIA__c where Status__c = 'Invoiced' limit 150];
list<string> testme = new list<string>{};
for(PreferredIA__c a:tobeclosed) {
testme.add(a.Account__r.id);
}
InvoicedAccounts = [select id, EligibleIAs__c, PreferredOverride__c,
Preferred_Territory__r.rep__c, LastSurveyDate__c,
InitialInspectionComplete__c, Program_level__c,
PreferredExempt__c, Account_Status__c,
Active_IAs__c, Last_Training__c
from Account where id IN :testme];
Contactlists = [select id, Account__c
from PreferredContact__c where Account__c IN :testme];
for(PreferredIA__c q:tobeclosed) {
q.Status__c = 'Closed';
}
for(Account z:invoicedaccounts) {
/****************************************************************
The following condition is where I am trying to set the z.EligibleIAs__c
which is a roll up count field of PreferredIA__c objects associated with
the account.
****************************************************************/
if(z.EligibleIAs__c == 0
&& z.Program_Level__c == 'Preferred'
&& !z.PreferredExempt__c
&& (z.Account_Status__c == 'Active'
|| z.Account_Status__c == 'Product Only')) {
loopadd = new PreferredIA__c();
system.debug(z.id);
system.debug(z.Account_Status__c);
loopadd.Account__c = z.id;
if(z.PreferredOverride__c != null) {
loopadd.Specialist__c = z.PreferredOverride__c;
}
else {
loopadd.Specialist__c= z.Preferred_territory__r.Rep__c;
}
for(PreferredContact__c q:contactlists) {
if(q.Account__c == z.id) {
loopadd.PreferredContact__c = q.id;
}
}
loopadd.CreatedDate__c = Date.Today();
if(z.Last_training__c != null) {
loopadd.DueDate__c = z.Last_Training__c.AddDays(365);
}
else {
loopadd.DueDate__c = Date.Today().AddDays(365);
}
loopadd.initial__c = false;
loopadd.Status__c = 'Unacknowledged';
newPreIAs.add(loopadd);
}
z.InitialInspectionComplete__c = true;
}
try {
update tobeclosed;
update invoicedaccounts;
insert newPreIAs;
}
catch(system.dmlexception q) {
system.debug(q);
system.debug(invoicedaccounts);
system.debug(newPreIAs);
}
}
public void ReceivePPW() {
monkey = [select id, Status__c from PreferredIA__c
where id in :preferredias and status__c = 'Training Completed'];
for (PreferredIA__c m:monkey) {
m.status__c = 'Awaiting Invoice';
}
update monkey;
}
}

I can't actually see where you're trying to write to the field — or did you remove it because it wasn't working?
That aside, the answer is that you can not write to a roll-up summary field. If you require a value in that field you should insert child records to your parent test records, with appropriate field values such that your summary field calculates a value.
Also, I can see that you're querying PerferredIA__c at the start, your test methods should never depend on data being in the system already, you should insert your records yourself in your test code. The reason for this is that if you try to deploy to an org which has no relevant data, your tests will fail and so, subsequently, will your deployment.

For situations like these, consider mock objects (or just variables simulating expected values), similar to inserting test values as Lacey suggests. This technique is required to achieve 100% coverage when doing callouts, for example, since they terminate tests at the moment of the call.

Related

Create Test Class for After Insert/After Update

this is my first question here.
Nice to meet you all, and looking forward to be here with you from now onward.
I am relatively new to coding with apex in general.
And I need to create the test class for the following code:
trigger CPTC on ContactPointTypeConsent (after insert, after update)
{
for (ContactPointTypeConsent c : Trigger.new)
{
Id ii = [Select PartyId From ContactPointTypeConsent WHERE Id = :c.Id].PartyId;
List<ContactPointTypeConsent> cptcL = [Select id, (Select id, Name, PrivacyConsentStatus From Individuals)
From Individual Where id = :ii].Individuals;
List<Contact> contacts = [Select id, (Select id, Name, MarketingSubscription__c From Contacts)
From Individual Where id = :ii].Contacts;
List<Lead> leads = [Select id, (Select id, Name, MarketingSubscription__c From Leads)
From Individual Where id = :ii].Leads;
Boolean checkBoxValue = checkStatus(cptcL);
for (Contact contact : contacts)
{
contact.MarketingSubscription__c = checkBoxValue;
}
for (Lead lead : leads)
{
lead.MarketingSubscription__c = checkBoxValue;
}
update contacts;
update leads;
}
public Boolean checkStatus(List<ContactPointTypeConsent> cptc)
{
Boolean result = false;
for (ContactPointTypeConsent c : cptc)
{
if (c.PrivacyConsentStatus == 'OptIn')
{
result = true;
Break;
}
}
return result;
}
}
Thank you all for your help, and time!

How to change the schema at runtime in Dapper-Extensions?

I am using Dapper Extensions but I have multiple schema names in My DB.
I found the an answer in below link but it assumes that I have only one schema name and this is not my case.
Dapper Extensions Change Schema
What is the right way to change the schema name at the runtime?
Probably you won't use it anymore, but maybe there are more guys up there trying to figure it out.
I've been checking out how dapper works when running GetAll<> and Insert<>.
What it does:
var type = typeof(T);
var cacheType = typeof(List<T>);
if (!GetQueries.TryGetValue(cacheType.TypeHandle, out string sql))
{
GetSingleKey<T>(nameof(GetAll));
var name = GetTableName(type); <--- key thing
sql = "select * from " + name;
GetQueries[cacheType.TypeHandle] = sql;
}
So I've checked what GetTableName was doing:
private static string GetTableName(Type type)
{
if (TypeTableName.TryGetValue(type.TypeHandle, out string name)) return name;
if (TableNameMapper != null)
{
name = TableNameMapper(type); <-- key thing
}
else
{
//NOTE: This as dynamic trick should be able to handle both our own Table-attribute as well as the one in EntityFramework
var tableAttr = type
#if NETSTANDARD1_3
.GetTypeInfo()
#endif
.GetCustomAttributes(false).SingleOrDefault(attr => attr.GetType().Name == "TableAttribute") as dynamic;
if (tableAttr != null)
{
name = tableAttr.Name;
}
else
{
name = type.Name + "s";
if (type.IsInterface() && name.StartsWith("I"))
name = name.Substring(1);
}
}
TypeTableName[type.TypeHandle] = name;
return name;
}
Solution:
So I thought I can implement my own table name mapper with code like this:
SqlMapperExtensions.TableNameMapper = DapperMapper.TableNameMapper();
And
public static SqlMapperExtensions.TableNameMapperDelegate TableNameMapper()
{
return (type) =>
{
var has = Attribute.GetCustomAttribute(type, typeof(CastleTableAttribute));
if (has != null)
{
return $"{ConfigurationProvider.Schema}.{type.Name}";
}
else
{
return type.Name;
}
};
}
With that name mapper you just need to mark your table with that.
[CastleTable]
class CubeTimestamps
{
[ExplicitKey]
public int cube_id { get; set; }
public DateTime cube_timestamp { get; set; }
}
you can implement your name mapper to use [TableName] attribute also. Because my simple implementation hides this feature.
Enjoy :)

Get the database ID (IDENTITY COLUMN) of newly created entity before context.SaveChanges()

Is it possible to reserve an IDENTITY value for a to-be-added row in the database using Entity Framework?
In my override of ObjectContext.SaveChanges, I want to do something like this:
public override int SaveChanges(SaveOptions options)
{
var affectedEntities = ObjectStateManager.GetObjectStateEntries(
EntityState.Added |
EntityState.Modified |
EntityState.Deleted);
foreach (var entry in affectedEntities)
{
var type = GetObjectType(entry.Entity.GetType());
if (entry.State == EntityState.Added)
{
// log the creation of the new entity into a log file
// or a different table. For this, I need the newly
// created entity's ID. Is it possible?
}
}
return base.SaveChanges(options);
// if I check for ObjectState here, i.e. after SaveChanges,
// it won't work? Because once the changes are committed to the database
// the newly added object's state would no longer remain
// Added.
}
You can't ask the database to preallocate the id's. You have 2 options
implement your own identity server
get the id's after the data has been written to the database
Here's a couple of examples of option 2. They both do the same thing - build a list of added items before the call to SaveChanges() and iterate over them after.
public interface IIDentifiable
{
int id { get; set; }
}
public override int SaveChanges()
{
ChangeTracker.DetectChanges();
List<IIDentifiable> addedObjects = GetNewEntries();
int result = base.SaveChanges();
foreach (IIDentifiable o in addedObjects)
{
//log o.id
}
return result;
}
private List<IIDentifiable> GetNewEntries()
{
var result = (
from e in ChangeTracker.Entries()
where e.State == System.Data.EntityState.Added
select e.Entity)
.OfType<IIDentifiable>()
.ToList();
return result;
}
the drawback of IIDentifiable is getting other details such as Type etc.
The second option is to do the same thing but use dynamic instead of IIDentifiable
public override int SaveChanges()
{
ChangeTracker.DetectChanges();
List<dynamic> addedObjects = GetNewEntries();
int result = base.SaveChanges();
foreach (dynamic o in addedObjects)
{
//log o.id
//log o.GetType()
}
return result;
}
private List<dynamic> GetNewEntries()
{
var result = (
from e in ChangeTracker.Entries()
where e.State == System.Data.EntityState.Added
select e.Entity)
.ToList();
return result;
}

Salesforce.com Apex Test Class issue: Need to hard code Opportunity Custom Field Value

I have a VF page form that has a text field. The value of this text field is compared to an existing opportunity custom field. If the value of this text field exists in an opportunity, I go ahead and create a new record.
Everything works out fine except when I create a Apex Test Class.
I need to create an opportunity and hard code the value for the custom field. Unfortuneatly, this field is not writeable.
Has anyone ran into this issue before?
This is the error that I'm getting:
Error: Compile Error: Field is not writeable: Opportunity.Deal_Registration_ID__c at line 9 column 103
Below is my Apex Class.
My test class is at the very bottom.
Any help is greatly appreciated.
Cheers!
Rommel
public with sharing class VaultVF {
public Vault__c vault {get; set;}
public List<String> errorMsgs {get; set;}
public String saveResult {get; set;}
public String dealReg {get; set;}
public String oppId {get; set;}
public String email {get; set;}
public Boolean scenario1{get; set;}
public Boolean scenario2{get; set;}
public Boolean scenario3{get; set;}
// Generates country dropdown from country settings
public List<SelectOption> getCountriesSelectList() {
List<SelectOption> options = new List<SelectOption>();
options.add(new SelectOption('', '-- Select One --'));
// Find all the countries in the custom setting
Map<String, Country__c> countries = Country__c.getAll();
// Sort them by name
List<String> countryNames = new List<String>();
countryNames.addAll(countries.keySet());
countryNames.sort();
// Create the Select Options.
for (String countryName : countryNames) {
Country__c country = countries.get(countryName);
options.add(new SelectOption(country.Two_Letter_Code__c, country.Name));
}
return options;
}
//Constructor, set defaults
public VaultVF(ApexPages.StandardController controller) {
vault = (vault__c)controller.getRecord();
//Populate list of random characters for CAPTCHA verification
characters = new List<String>{'a','b','c','d','e','f','g','h',
'i','j','k','m','n','p','q','r','s','t','u','v','w',
'x','y','z','1','2','3','4','5','6','7','8','9'
};
errorMsgs = new List<String>();
}
//Determine if page has error so errors message can be displayed
public boolean getHasErrors(){
if(ApexPages.hasMessages() == true){
if(errorMsgs == null){
errorMsgs = new List<String>();
}
//Loop through errors and add to a list
for(ApexPages.Message m : ApexPages.getMessages()){
if(m.getSummary().startsWith('Uh oh.')){
errorMsgs.add(String.valueOf(m.getSummary()));
}
}
}
return ApexPages.hasMessages();
}
public void save(){
errorMsgs = new List<String>();
//Make sure captcha was correct
if(validateCAPTCHA() == false){
errorMsgs.add('Verification code was incorrect.');
captchaInput = '';
}
//Make sure the Scenario is selected
if(scenario1 == true){
vault.Scenario__c = 'BIG-IP as a Firewall';
}else if(scenario2 == true){
vault.Scenario__c = 'BIG-IP providing firewall service in conjunction with other vendors';
}else if(scenario3 == true){
vault.Scenario__c = 'BIG-IP front ending Firewalls';
if(vault.Load_Balancer_is_a_Firewall__c == null){
errorMsgs.add('Please specify if the load balancer is a firewall.');
}
if(vault.Providing_DDos_Connection__c == null){
errorMsgs.add('Please specify if this is providing DDos connection level protection for firewalls or both.');
}
if(vault.Providing_DDos_protection_except_ASM__c == null){
errorMsgs.add('Please specify if we are providing DDos protection (except ASM).');
}
if(vault.No_Traffic_Traverse__c == null){
errorMsgs.add('Please specify if there is Traffic that does not traverse the Firewalls.');
}
if(vault.Firewall_Vendor_Name__c == null){
errorMsgs.add('Please specify a Firewall Vendor.');
}
} else {
errorMsgs.add('Please select one of three Scenarios that is similar to your setup');
}
//-----Need to make sure deal registration number on opportuntiy is valid.-----
//Set some sort of is valid deal reg flag to false. We will assume the deal reg Id entered is invalid until we determine it is valid
vault.isValidDealReg__c = false;
//Query Account Id, and Deal_Registration_Id__c from the opportunity object where the deal registration Id equals the value entered in the form
List<Opportunity> opps = [select Id, Deal_Registration_ID__c from Opportunity where Deal_Registration_ID__c = :vault.Deal_Registration__c];
//check to see if registration id on the opp match the entered value
if(opps.size() > 0 && opps[0].Deal_Registration_ID__c == vault.Deal_Registration__c){
//If they do match set the valid deal reg Id flat to true
vault.isValidDealReg__c = true;
vault.Related_Opp__c = opps[0].Id;
//If they don't match then query all contacts on the account related to the opportunity
}
//If is valid deal reg flag is false add a message to the errorMSgs list indicated the entered
//deal registration is not valid with details on who they should contact
if(errorMsgs.size()>0){
//stop here
} else if(vault.isValidDealReg__c == false){
errorMsgs.add('Deal Registration # was incorrect. Please contact your representative.');
} else {
try{
vault.Status__c = 'Application Received';
insert vault;
saveResult = 'success';
}catch(exception e){
ApexPages.addMessage(new ApexPages.Message(ApexPages.Severity.FATAL, 'Uh oh. Something didn\'t work quite right. Please contact partners#f5.com for assistance. \n\n' + e.getMessage()));
saveResult = 'fail';
}
}
}
//------------------------Code for CAPTCHA verification----------------------------
public String captchaInput {get; set;}
List<String> characters;
String char1;
String char3;
String char5;
//This methods simply returns a random number between 0 and the size of the character list
public Integer randomNumber(){
Integer random = Math.Round(Math.Random() * characters.Size());
if(random == characters.size()){
random--;
}
return random;
}
/*Here we have 6 get methods that return 6 random characters to the page.
For chars 1,3, and 5 (the black characters) we are saving the the values so
that we can compare them with the user's input */
public String getChar1(){
char1 = characters[randomNumber()];
return char1;
}
public String getChar2(){
return characters[randomNumber()];
}
public String getChar3(){
char3 = characters[randomNumber()];
return char3;
}
public String getChar4(){
return characters[randomNumber()];
}
public String getChar5(){
char5 = characters[randomNumber()];
return char5;
}
public String getChar6(){
return characters[randomNumber()];
}
public String correctAnswer(){
return char1 + char3 + char5;
}
public Boolean validateCAPTCHA(){
if(captchaInput.length() == 3 && captchaInput.subString(0,1) == char1 && captchaInput.subString(1,2) == char3 && captchaInput.subString(2,3) == char5){
return true;
}else{
return false;
}
}
}
Below is Test Class
#isTest
public class VaultVFTEST {
/*This tests the VaultVF class on the Vault__c object*/
static testMethod void VaultVFTEST() {
//Create opp, hardcode the Deal_Registration_Id__c field with 'x111111111' and insert opp
Opportunity opp = new Opportunity(Name='test opp', StageName='stage', Deal_Registration_Id__c='x111111111', Probability = 95, CloseDate=system.today());
insert opp;
//Create and insert Vault record
Vault__c vault = new Vault__c(
Deal_Registration__c = 'x111111111',
Scenario__c = 'BIG-IP front ending Firewalls',
isValidDealReg__c = true,
Status__c = 'Application Received',
Load_Balancer_is_a_Firewall__c = 'Yes',
Providing_DDos_Connection__c = 'Firewalls',
Providing_DDos_protection_except_ASM__c = 'Yes',
No_Traffic_Traverse__c = 'Yes',
Firewall_Vendor_Name__c = 'Firewall Vendor Company',
First_Name__c = 'Test First Name',
Last_Name__c = 'Test Last Name',
Company_Name__c = 'Test Company Name',
Phone__c = '(206) 946-3126',
Email__c = 'test#email.com',
Country__c = 'United States',
Additional_Info__c = 'Test Additional Info',
Related_Opp__c = opp.Id
);
//Now insert the vault record to invoke the VaultVF class
Test.startTest();
insert vault;
Test.stopTest();
//Assertion Testing - Check to make sure the Deal_Registration__c value
//matches the Deal_Registration_Id__c value of the related opportunity
for(Vault__c v : [select Id, Deal_Registration__c from Vault__c where Deal_Registration__c IN :vault]){
system.assertEquals(v.Deal_Registration__c, opp.Deal_Registration_Id__c);
}
}
}
What's the Opportunity.Deal_Registration_ID__c field type? If it's a formula field or autonumber it will really really be not writeable from Apex.
Generally speaking in such scenarios you'd have to insert the Opportunity (without this field set), then read it back from database after a successful insert. If it's autonumber - magic will happen on it's own (although I wouldn't say it's a best idea to have some kind of license numbers consecutive & easy to fake ;)), if it's a formula field - make sure you've set all fields that are used in calculation prior to the insert.
So something like this should do the trick:
Opportunity opp = new Opportunity(Name='test opp', StageName='stage', Probability = 95, CloseDate=system.today());
insert opp;
opp = [SELECT Id, Name, Deal_Registration_ID__c FROM Opportunity WHERE Id = :opp.Id];
// Optionally - make sure value was set
System.assertNotEquals(null, opp.Deal_Registration_ID__c);
System.assert(opp.Deal_Registration_ID__c.length() > 0); // if it's a String
Vault__c vault = new Vault__c(
Deal_Registration__c = opp.Deal_Registration_ID__c ,
Scenario__c = 'BIG-IP front ending Firewalls',
isValidDealReg__c = true,
Status__c = 'Application Received',
Load_Balancer_is_a_Firewall__c = 'Yes',
Providing_DDos_Connection__c = 'Firewalls',
Providing_DDos_protection_except_ASM__c = 'Yes',
No_Traffic_Traverse__c = 'Yes',
Firewall_Vendor_Name__c = 'Firewall Vendor Company',
First_Name__c = 'Test First Name',
Last_Name__c = 'Test Last Name',
Company_Name__c = 'Test Company Name',
Phone__c = '(206) 946-3126',
Email__c = 'test#email.com',
Country__c = 'United States',
Additional_Info__c = 'Test Additional Info',
Related_Opp__c = opp.Id
);

automatically insert in a master relationship field

I have a VF page with a master relationship field, can I insert value in this field automatically
<apex:inputField value="{!a.Parent__c}"/>
Question is a bit vague but if you want to set a default value for the Parent__c you can just apply the value in your controller class.
EXAMPLE
public class VFController {
private Account controllerAccount;
//Basic Constructor
public VFController() {
controllerAccount = new Account();
controllerAccount.Parent__c = getDefaultParentAccount();
}
//Standard Controller Constructor
public VFController(ApexPages.Controller standardController) {
controllerAccount = (Account)standardController.getRecord();
controllerAccount.Parent__c = getDefaultParentAccount();
}
private Id getDefaultParentAccount() {
Id parentAccountId = null;
//Lookup default parent Account
List<Account> parentAccounts = [SELECT Id FROM Account WHERE /*Your Criteria here*/ LIMIT 1];
if (parentAccounts.isEmpty() == false) {
parentAccountId = parentAccounts[0].Id;
}
return parentAccountId;
}
}

Resources