Salesforce CRM's REST API allows for inserting up to 200 records into Salesforce using just a single API call, a considerable time- and cost-savings over inserting records one-by-one. I've updated my Java Salesforce client to be able to make such calls. Some notes about this process:
This multiple insertion process is distinct from Salesforce's Bulk API, an asynchronous method that relies on CSV files that is suggested for data sets of over 2000 records where immediate responses are not necessary.
This process is just for record insertions. For other CRUD actions, Salesforce's Composite API can be used, albeit with lower limits (25 requests per query).
Salesforce provides a multi-insert example showing the request and successful (201 Created) response JSON, but is missing the format of error messages specific to these types of calls. I'm providing below the format of the error responses determined while updating the Java client. Given a snippet of the SF example's request body:
{ "records" :[{ "attributes" : {"type" : "Account", "referenceId" : "ref1"}, "name" : "SampleAccount1", "phone" : "1111111111", "website" : "www.salesforce.com", "numberOfEmployees" : "100", "industry" : "Banking" },{ "attributes" : {"type" : "Account", "referenceId" : "ref2"}, "name" : "SampleAccount2", "phone" : "2222222222", "website" : "www.salesforce2.com", "numberOfEmployees" : "250", "industry" : "Banking" },... ] }
Each item to be inserted needs an attributes
metadata property, specifying the type
and a referenceId
, the latter of which can be anything but must be unique for each item in the insertion. The reference IDs are used in the response for either providing the Salesforce IDs for successful insertions, or in referring to any errors with that particular record. The attribute type
field seems redundant, as the API call made already specifies the type being inserted, but it is nonetheless required for these types of calls.
Generic error response (403) if any reference IDs are missing:
[ { "message": "Include a reference ID for each record in the request.", "errorCode": "INVALID_INPUT" } ]
Above message will be uncommon so long as the request has referenceIds for every record provided. The more common 400 Bad Request response can occur due to missing attribute types, duplicate reference IDs, as well as validation failures, missing required fields, etc. For 400s, the response body will list the problem records by the referenceId provided in the request, example:
{ "hasErrors": true, "results": [ { "referenceId": "ref3", "errors": [ { "statusCode": "INVALID_INPUT", "message": "Duplicate ReferenceId provided in the request.", "fields": [] } ] }, { "referenceId": "ref8", "errors": [ { "statusCode": "INVALID_INPUT", "message": "Include an entity type for each record in the request.", "fields": [] } ] } ] }
What is important to note with multiple record insertion is that, if there are any reported problems with any of the items being inserted, none of the records in the request will be inserted (all-or-nothing). One way to handle failures is to make a second request of the same records minus those reported as failures in the prior response (matching on referenceId), to at least get those records inserted. The failed records can instead be logged and analyzed to see what to do with them.
As for making these calls using the Salesforce Client, an included integration test shows the process for inserting multiple rows with one call, and also how to trap and read any 400 exceptions that may occur. The code is fairly the same regardless of which objects are being inserted, however each type of object being inserted will need a MultipleEntityRecord subclass (similar to here for the integration test). The MultipleEntityRecord base class stores the required type
and referenceId
attributes, while the subclass is to store the fields specific to the object being inserted.
Posted by Glen Mazza in Salesforce CRM at 02:00AM Nov 14, 2022 | Tags: salesforce salesforce-crm | Comments[0]
This post provides SOAP and Java examples of activating list sends in Marketing Cloud (see my earlier post for the alternative of triggered sends). In this situation, we provide the email contents along with the MC-stored subscriber list(s) to send the email to.
For SOAP, one creates a Send object which wraps (among other values) an Email, EmailSendDefinition and one more more subscriber List objects. Some of the more important values stored at each level:
Object | Information to provide within object |
---|---|
Send | Wrapper for below three objects, also stores the email from-address and from-name. |
HTML and text versions of the email, subject line, and character set. See some examples. | |
List | MC List IDs to send the email to. List IDs are available from MC Email Studio, menu item Subscribers | Lists, selecting the list and viewing its Properties tab. |
EmailSendDefinition | Whether or not to use multipart emails, to send de-duplicate (not to send multiple copies to the same email address if the address is on multiple lists that the email is being sent to). The default values, need to provide them, and whether MC actually does anything with certain properties aren't always clear, you will probably need to experiment a bit. |
Here's a SOAP example using the MC Postman workspace:
<?xml version="1.0" encoding="UTF-8"?> <s:Envelope xmlns:s="http://www.w3.org/2003/05/soap-envelope" xmlns:a="http://schemas.xmlsoap.org/ws/2004/08/addressing" xmlns:u="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd"> <s:Header> <a:Action s:mustUnderstand="1">Create</a:Action> <a:To s:mustUnderstand="1">https://{{et_subdomain}}.soap.marketingcloudapis.com/Service.asmx</a:To> <fueloauth xmlns="http://exacttarget.com">{{dne_etAccessToken}}</fueloauth> </s:Header> <s:Body xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> <CreateRequest xmlns="http://exacttarget.com/wsdl/partnerAPI"> <Objects xsi:type="Send"> <Client> <ID>{{et_mid}}</ID> </Client> <Email> <Name>Sample Email Send</Name> <IsHTMLPaste>true</IsHTMLPaste> <Subject>Sample Test Message</Subject> <CharacterSet>UTF-8</CharacterSet> <HTMLBody>Testing message: %%[ if listid != 1234567 then ]%% Welcome reader! %%[else]%% Greetings reader! %%[endif]%%</HTMLBody> <TextBody>Welcome Reader! (text only)</TextBody> </Email> <List> <ID>1234567</ID> </List> <List> <ID>2345678</ID> </List> <EmailSendDefinition> </EmailSendDefinition> <FromAddress>bobsemail@yopmail.com</FromAddress> <FromName>Bob Sender</FromName> </Objects> </CreateRequest> </s:Body> </s:Envelope>
The "Name" field with value "Sample Email Send" does not appear in the email but is used to help identify a specific email send. It is what is displayed in the Sends section of the Email Studio home page and the also the Tracking section of the list details. It does not have to be unique (sends are identified by a unique Job ID) but making it so helps make sends easier to tell apart from each other.
Sometimes you may wish to adjust the email a bit depending on the list being sent to. In the HTMLBody element of the above example, I've added AMPScript tags showing how this can be done.
A sample SOAP response for a request as above is as follows. Note the response contains the Job ID and the request ID that SFMC generated for your request:
<?xml version="1.0" encoding="utf-8"?> <soap:Envelope xmlns:soap="http://www.w3.org/2003/05/soap-envelope" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsa="http://schemas.xmlsoap.org/ws/2004/08/addressing" xmlns:wsse="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd" xmlns:wsu="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd"> <env:Header xmlns:env="http://www.w3.org/2003/05/soap-envelope"> <wsa:Action>CreateResponse</wsa:Action> <wsa:MessageID>urn:uuid:5ae72a13-31ac-4cc2-866e-50d8407fa5fe</wsa:MessageID> <wsa:RelatesTo>urn:uuid:93d44d6d-c424-4213-927e-3f1aac085b8b</wsa:RelatesTo> <wsa:To>http://schemas.xmlsoap.org/ws/2004/08/addressing/role/anonymous</wsa:To> <wsse:Security> <wsu:Timestamp wsu:Id="Timestamp-6987861d-c008-402f-ad7c-d30dae59b1d3"> <wsu:Created>2023-11-04T02:02:11Z</wsu:Created> <wsu:Expires>2023-11-04T02:07:11Z</wsu:Expires> </wsu:Timestamp> </wsse:Security> </env:Header> <soap:Body> <CreateResponse xmlns="http://exacttarget.com/wsdl/partnerAPI"> <Results> <StatusCode>OK</StatusCode> <OrdinalID>0</OrdinalID> <NewID>9928401</NewID> </Results> <RequestID>e893e94c-4b2d-4a9d-ad12-71eb8d62e073</RequestID> <OverallStatus>OK</OverallStatus> </CreateResponse> </soap:Body> </soap:Envelope>
For Java, an example using the Fuel SDK is below.
public void sendEmail() { CreateRequest createRequest = new CreateRequest(); CreateOptions createOptions = new CreateOptions(); createOptions.setRequestType(RequestType.SYNCHRONOUS); createOptions.setQueuePriority(Priority.HIGH); createRequest.setOptions(createOptions); Send send = new Send(); com.exacttarget.fuelsdk.internal.Email email = new com.exacttarget.fuelsdk.internal.Email(); email.setName("Sample Email Send via Java"); email.setEmailType(EmailType.HTML.value()); email.setIsActive(Boolean.TRUE); email.setIsApproved(Boolean.TRUE); email.setIsHTMLPaste(Boolean.TRUE); email.setSubject("Sample test message subject"); email.setCharacterSet("UTF-8"); email.setHtmlBody("<p>Email Body</p>"); email.setTextBody("Text version of email body"); send.setEmail(email); // add as many lists as needed List listToSendTo = new List(); listToSendTo.setId(1234567); send.getList().add(listToSendTo); // More on EmailSendDefinition: // https://developer.salesforce.com/docs/marketing/marketing-cloud/guide/creating_an_email_send_definition_using_the_web_service_api.html EmailSendDefinition emailSendDefinition = new EmailSendDefinition(); emailSendDefinition.setIsMultipart(isMultipart); emailSendDefinition.setDeduplicateByEmail(true); send.setEmailSendDefinition(emailSendDefinition); send.setFromAddress("bobsemail@yopmail.com"); send.setFromName("Bob Sender"); createRequest.getObjects().add(send); // configure ETClient similar to here: https://salesforce.stackexchange.com/a/312178 // ETClient etClient = .... CreateResponse response = etClient.getSoapConnection().getSoap().create(createRequest); if (response != null && "OK".equalsIgnoreCase(response.getOverallStatus())) { // success! Check email inbox... LOGGER.info("Success sending email w/Request ID {}", response.getRequestID()); } else { Optional.ofNullable(response) .ifPresent(cr -> Optional.ofNullable(cr.getResults()) .filter(errorList -> !errorList.isEmpty()) .map(errorList -> errorList.get(0)) .ifPresent(createResult -> { LOGGER.error("{}: {}", createResult.getErrorCode(), createResult.getStatusMessage()); } ) ); } }
Further Reading
Posted by Glen Mazza in Marketing Cloud at 02:00AM Nov 10, 2022 | Comments[0]