Develop Functions

This section explains the basic developer workflow when developing a new function. This section also details how to add build and run-time configurations for your functions.

When you're done developing your function, deploy your function to a Salesforce compute environment and invoke your function from your org.

To create a function, start by creating a Salesforce DX project.

The DX project name you choose represents the Salesforce Functions project name, which you use when invoking a deployed function. A Salesforce Functions project name must be unique to the org. The project name is listed in the sfdx-project.json configuration file.

Next, use the generate:function command to populate your project with the template source files and the metadata file for your function. For example, to create a JavaScript-based function named myfunction, use the following command from your DX project root directory:

The function name must be only lowercase letters or numbers, and must start with a lowercase letter.

The only supported languages for the -l/--language arguments are javascript, typescript, and java.

Creating a function adds the <project root>/functions/<function name> directory to your DX project and creates default source files and build definition files for the function in that directory. Here’s an example DX project with function-specific files created for a JavaScript function named myfunction:

For a step-by-step guide to creating a Functions project, see Quick Start.

Each function must provide a list of dependencies that the function must have to build and run. Function dependency information is provided in language-specific configuration files.

JavaScript or TypeScript functions specify dependencies and build configuration information in a Node.js package.json file in the <project root>/functions/<function name>/ directory. This file is required to build and run your JavaScript or TypeScript function. For details on the format and fields for a Node.js package.json file, see https://nodejs.org/docs/latest/api/packages.html#nodejs-packagejson-field-definitions.

Using npm install with the Node.js package.json is recommended when you start developing a function, which installs dependencies to enable things like type support for TypeScript functions code in VS Code. You can also use npm build to verify you've added the right set of dependencies.

Generated JavaScript and TypeScript functions include the Salesforce Functions Runtime for Node.js in the package.json file:

Dependencies are defined in your function's pom.xml file for use with the Maven build tool. The following pom.xml file includes the Salesforce SDK for Java Functions in the list of dependencies:

For more information on Maven and pom.xml files, see https://maven.apache.org/.

Each function must provide a project.toml TOML file that contains function metadata. This file usually resides in your <project root>/functions/<function name> directory. The generate:function command creates a template project.toml file that looks something like this:

On creation, salesforce-api-version reflects the current version of the Salesforce REST API that the Functions SDK uses. Salesforce Functions supports API version 53.0 or later.

Update this TOML file with any function metadata information you need. For a list of the valid fields for project.toml see Function Metadata TOML Files. For details on the general TOML file format, see toml.io.

In your function code, import the Salesforce Functions SDK for your programming language.

When you generate a function with sf generate function, these dependencies are included in the example code.

There's no Salesforce Functions SDK required when writing Salesforce Functions code in JavaScript.

In TypeScript, include the Salesforce Functions SDK for Node.js with import:

In Java, import classes that you need from the com.salesforce.functions.jvm.sdk package, for example:

Functions that access Salesforce data must have a specific code entry point that can be invoked with the invoking org’s context data and payload.

In JavaScript and TypeScript, specify and export a execute entry point. The following JavaScript example provides an execute entry point function:

The following TypeScript example provides an execute entry point function:

Java functions must provide a public class that implements the SalesforceFunction interface and overrides the public apply() method. The following example provides an implementation of apply() using FunctionInput and FunctionOutput classes defined elsewhere in the function project code:

The types supported for the input and output for SalesforceFunction are specific to the Java SDK and are described in Java Functions.

Salesforce Functions require that source code is tracked with git. When developing a function, add your project to a GitHub repo and push your function code changes regularly to collaborate with other developers. Functions code, unlike Apex code, doesn't get deployed to your org, so you can't use your orgs as a way to share code.

Function code must be committed to git before you can deploy a function. However, merging your function code to a github.com remote repo isn't required, although generally a good practice.

To add your project to GitHub, navigate to github.com in your browser. Log in to your github.com account, and create a repository. Save the git URL for your new repo. See Create a Repo for more details.

In the DX project root directory, use the following git commands:

Connect to additional Salesforce resources using the Salesforce Functions SDK for your language (Node.js or Java). The Salesforce Functions SDKs provide an integrated programming model for writing business logic that connects with your data in the Salesforce Platform.

The SDKs provide context data for the calling org when your function is invoked. The data is passed as a parameter to your function entry point. Through this context you can query and execute DML on your org data. Record access is controlled using the Functions permission set in your org.

Through the context data you can access the DataApi interface to query, insert, and update records.

The following JavaScript example uses context.org.dataApi to make a simple query to the org that invoked the function:

The following Java example uses the DataApi class from the context data to do a query:

For more complex access, such as complex or large transactions, the Salesforce Functions SDKs provide the UnitOfWork class. A UnitOfWork represents a set of one or more Salesforce operations that must be done as a single atomic operation. Single atomic operations reduce the number of requests back to the org, and is more efficient when working with larger data volumes. UnitOfWork also lets you manage data operations in your own transactions.

The following JavaScript example (from the Context_UnitOfWork_JS sample) uses a UnitOfWork to create an Account record and related records:

When using the Node.js SDK only, always use a new instance of UnitOfWork for each transaction and never reuse a committed UnitOfWork.

The following Java example (from the Context_UnitOfWork_Java sample) uses the UnitOfWork class from the Salesforce SDK for Java Functions (with Input and Output defined elsewhere in the function code):

UnitOfWork uses the Composite Graph API for efficient transaction requests with higher record limits. For more details on the Composite Graph API see: REST API Developer Guide: Composite Graphs. Note the (Composite Graph API limits)[https://developer.salesforce.com/docs/atlas.en-us.api_rest.meta/api_rest/resources_composite_graph_limits.htm], such as a maximum of 15 different nodes or objects or both in one payload, also apply to UnitOfWork.

If the provided SDK classes don't give you the data access you need, you can try making REST API calls directly to the calling org.

In the Salesforce SDK for Node.js Functions, you can use context.org.dataApi.accessToken to obtain the API access token for the invoking org. This token can be used with your preferred HTTP request framework to make REST API calls back to the invoking org. You can also use the token to initialize a JSForce connection to access these APIs:

In the Salesforce SDK for Java Functions, you can use DataApi.getAccessToken() to obtain the API access token for the invoking org. Use your API access token with your preferred HTTP request framework to make REST API calls back to the invoking org.

Most org access from functions is similar to an API request to an org through something like the Salesforce REST API, and has similar limits. See Limits.

The commands run:function:start and project:deploy use a specific set of buildpacks to build the container image for your function.

Deployed functions currently run in the Heroku-22 environment.

Learn how you can Use Heroku Data in Functions.