lyonstahl / salesforce
A library that provides a simple and intuitive way to interact with Salesforce API
Requires
- php: >=7.3
- guzzlehttp/guzzle: ^7.8
Requires (Dev)
- phpunit/phpunit: ^9.6
This package is auto-updated.
Last update: 2024-12-12 14:44:18 UTC
README
Installation
Ensure you have composer installed, then run the following command:
composer require lyonstahl/salesforce
That will fetch the library and its dependencies inside your vendor folder.
If you want a simple API for bulding SOQL queries, we suggest lyonstahl/soql-builder
Requirements
Features
- OAuth with password grant type
- Create, update, delete, upsert, and query records
- Object representation of Salesforce records with LyonStahl\Salesforce\Record
- Field validation and object mapping
- Extendable code: add custom objects or validation rules
Setting up a Connected App
Before you begin, you need to have set up a "Connected App" in Salesforce and get a consumerKey
, consumerSecret
, username
, and password
to allow Api access.
Note, this process may vary depending on changes to the Salesforce website, or on your Salesforce account and settings.
Check Salesforce's Help Docs to verify.
- Log into to your Salesforce org
- Click on Setup in the upper right-hand menu
- Under Build click Create → Apps
- Scroll to the bottom and click "New" under Connected Apps.
- Enter the following details for the remote application:
- Connected App Name
- API Name
- Contact Email
- Under the API dropdown, enable OAuth Settings
- Callback URL
- Select Access Scope (If you need a refresh token, specify it here)
- Click Save, and store your access credentials in a safe place.
Usage
Creating a new Api client and connecting:
use LyonStahl\Salesforce\Authenticator\Password; use LyonStahl\Salesforce\Client; // Provide endpoint and any Guzzle options in Password::create(), endoint defaults to login.salesforce.com $auth = Password::create(['endpoint' => 'https://test.salesforce.com/'])->authenticate([ "client_id" => $YOUR_CONSUMER_KEY, "client_secret" => $YOUR_CONSUMER_SECRET, "username" => $YOUR_SALESFORCE_USERNAME, "password" => $YOUR_SALESFORCE_PASSWORD_AND_SECURITY_TOKEN ]); // Here you may also provide object mappings and desired Salesforce API version $salesforce = new Client($auth, [], 59);
The following examples use a Salesforce object named Example
that has a field Name
.
Executing Basic SOQL Queries:
$select = "SELECT Id, Name FROM Example LIMIT 100"; foreach ($salesforce->query($select) as $object) { echo "Example {$object->Id} ({$object->Name})\n"; // outputs something like "Example 5003000000D8cuIQAA (Bob)" }
If you need to use php values in your query, put {tokens}
in your SOQL and pass the values separately via query()
's second argument. The values will be properly quoted and escaped based on their type, and interpolated into the query:
$select = "SELECT Id, Name FROM Example WHERE Name={name} LIMIT 100"; $name = 'Bob'; foreach ($salesforce->query($select, ['name' => $name]) as $object) { echo "Example {$object->Id} ({$object->Name})\n"; // outputs something like "Example 5003000000D8cuIQAA (Bob)" }
Fetching a Record by Id:
$id = "5003000000D8cuIQAA"; $bob = $salesforce->get("Example", $id); echo "Hello, {$bob->Name}\n"; // outputs something like "Hello, Bob"
Creating a new Record:
use LyonStahl\Salesforce\Record; $linda = $salesforce->create(new Record("Example", ["Name" => "Linda"])); echo "Example {$linda->Id} ({$linda->Name})"; // outputs something like "Example 5003000000D8cuIQAA (Linda)"
Updating an existing Record:
$bob->Name = "Roberto"; $roberto = $salesforce->update($bob); echo "Hello, {$roberto->Name}\n"; // outputs "Hello, Roberto"
Deleting a Record:
$ded = $salesforce->delete($bob); var_dump($ded->Id); // outputs "NULL"
Advanced Usage
Extending the Record class
The included LyonStahl\Salesforce\Record
class can be used without modification as a generic "salesforce record" implementation - it will automatically set properties based on what's fetched from the Api. However, the intent is that applications will extend from it and define the properties needed for each of their Salesforce objects. This allows for a consistent schema that your code can rely on, and even lets you implement some level of validation directly in your application.
To build your own Salesforce Object, you must:
- extend from
LyonStahl\Salesforce\Record
- define the object fields as public properties
- list any properties that must not be included in update() calls (e.g., renamed fields, nested objects or object lists) in UNEDITABLE_FIELDS.
- add any necessary logic in
setField()
(e.g., building a new object if your record has a relation) - add any desired logic in
validateField()
Using our "Example" object from above,
use LyonStahl\Salesforce\Record; class Example extends Record { public const TYPE = "Example"; public ?string $Name = null; }
Inline Records and Record Lists
SOQL allows queries for nested objects and queries, which appear in results as inline records and query results respectively. This library does understand results from such queries, but your Record subclasses must define properties in a particular way to support them.
For example, a query similar to SELECT Manager.Id, Manager.Name, (SELECT Id, Name FROM Members) FROM Teams
would require a Record class like so:
use LyonStahl\Salesforce\Record; use LyonStahl\Salesforce\Result; use Example; class Team extends Record { public const TYPE = "Team"; protected const UNEDITABLE_FIELDS = [ "Manager", "Members", ...parent::UNEDITABLE_FIELDS ]; public ?Example $Manager = null; public ?Result $Members = null; }
Where there is an inline object, the property should be typed as the corresponding Record subclass. For any record type where you're not making a Record subclass, type the property as Record
— though this is obviously less useful.
Where there is a subquery, the property should be typed as a Result
instance. Among other things, this allows the Result to support paginated subqueries.
Object Mapping
Finally, to allow the Api Client take advantage of your subclasses, you must provide an $objectMap
so it knows which PHP classes correspond to which Salesforce record types. Without this, you'll end up with generic Record instances for everything. Nested Results will be provided the same $objectMap
as their parent Result instance.
$salesforce = new Client( $password->authenticate([...$credentials]), [Example::TYPE => Example::class, Team::TYPE => Team::class] );
Field Validation
Some basic validation functions are included in LyonStahl\Salesforce\Validator
. These methods all take the value to validate as the first argument, and can have other arguments depending on needs. Follow this same pattern to implement additional validation functions for your own objects as needed.
use LyonStahl\Salesforce\Record; use LyonStahl\Salesforce\Validator; class Example extends Record { public ?string $Name = null; protected function validateField(string $field) : void { switch ($field) { case "Name": Validator::characterLength($this->Name, 2, 100); return; default: parent::validateField($field); return; } } }
Handling Errors
All Runtime Exceptions thrown from this library will be an instance of LyonStahl\Salesforce\Error
.
Exceptions are grouped into the following types:
-
LyonStahl\Salesforce\Exceptions\SalesforceException
:Errors originating from the Salesforce API, including HTTP errors (e.g., connection timeouts)
-
LyonStahl\Salesforce\Exceptions\AuthException
:Authentication failures or attempts to use the HttpClient before authentication has succeeded
-
LyonStahl\Salesforce\Exceptions\ResultException
:Errors parsing or handling Salesforce Api results or records; these will usually indicate a problem in your custom Record classes
-
LyonStahl\Salesforce\Exceptions\UsageException
:Errors arising from incorrect library usage; these will usually indicate a runtime problem in your application code
-
LyonStahl\Salesforce\Exceptions\ValidationException
:Validation errors.
Running for development with Docker
We have included a Dockerfile to make it easy to run the tests and debug the code. You must have Docker installed. The following commands will build the image and run the container:
docker build -t lyonstahl/salesforce --build-arg PHP_VERSION=8 .
docker run -it --rm -v ${PWD}:/var/www/sapi lyonstahl/salesforce sh
Debugging with XDebug in VSCode
Docker image is configured with XDebug. To debug the code with VSCode, follow these steps:
-
Install the PHP Debug extension in VSCode
-
Add a new PHP Debug configuration in VSCode:
{ "name": "XDebug Docker", "type": "php", "request": "launch", "port": 9003, "pathMappings": { "/var/www/sapi/": "${workspaceRoot}/" } }
-
docker run -it --rm -v ${PWD}:/var/www/sapi --add-host host.docker.internal:host-gateway lyonstahl/salesforce sh
-
Start debugging in VSCode with the 'XDebug Docker' configuration.
Testing
This library ships with PHPUnit for development. Composer file has been configured with some scripts, run the following command to run the tests:
composer test