codelicia / trineforce
Salesforce Soql Doctrine Driver
Installs: 10 895
Dependents: 0
Suggesters: 0
Security: 0
Stars: 5
Watchers: 2
Forks: 6
Open Issues: 12
pkg:composer/codelicia/trineforce
Requires
- php: ^8.1 || ^8.2
- ext-json: *
- ext-pdo: *
- beberlei/assert: ^v3.3.2
- doctrine/dbal: ^3.5.1
- doctrine/orm: ^2.13.3
- guzzlehttp/guzzle: ^6.0 | ^7.0
- guzzlehttp/psr7: ^2.4.3
- psr/http-message: ^1.0.1
Requires (Dev)
- doctrine/coding-standard: ^11.1.0
- maglnet/composer-require-checker: ^4.2.0
- malukenho/mcbumpface: ^1.1.5
- phpunit/phpunit: ^8.3 | ^9.0
- roave/security-advisories: dev-master
- staabm/annotate-pull-request-from-checkstyle: ^1.8.3
- 2.x-dev
- 1.2.x-dev
- 1.2.1
- 1.2.0
- 1.1.x-dev
- 1.1.1
- 1.1.0
- 1.0.x-dev
- 1.0.0
- 0.4.x-dev
- 0.4.0
- 0.3.x-dev
- 0.3.3
- 0.3.2
- 0.3.1
- 0.3.0
- 0.2.x-dev
- 0.2.0
- 0.1.0
- 0.0.10
- 0.0.9
- 0.0.8
- 0.0.7
- 0.0.6
- 0.0.5
- 0.0.4
- 0.0.3
- 0.0.2
- 0.0.1
- dev-renovate/doctrine-coding-standard-14.x
- dev-renovate/azjezz-psl-4.x
- dev-renovate/actions-checkout-5.x
- dev-renovate/phpunit-phpunit-12.x
- dev-renovate/infection-infection-0.x
- dev-renovate/doctrine-dbal-4.x
- dev-infection
- dev-updates
- dev-outdated-main
This package is auto-updated.
Last update: 2025-09-30 23:28:47 UTC
README
Salesforce Soql Doctrine Driver allows you to write Soql queries and interact with a Salesforce instance using the Doctrine DBAL layer.
Now one can forget about Salesforce and have a nice repository/query object
integration on one's architecture without hurting that much on the usual
project structure.
Installation
Use composer to install this package as bellow:
$ composer require codelicia/trineforce
Configuration
If you are familiar with Doctrine, then you probably already know how to configure and use it. But some special configuration is required in order to make it work.
When creating a new Connection, you should also provide the configuration
keys for salesforceInstance, consumerKey, consumerSecret and point to
the right driverClass. The usual user and password are also required.
$config = new Configuration(); $connectionParams = [ 'salesforceInstance' => 'https://[SALESFORCE INSTANCE].salesforce.com', 'apiVersion' => 'v43.0', 'user' => 'salesforce-user@email.com', 'password' => 'salesforce-password', 'consumerKey' => '...', 'consumerSecret' => '...', 'driverClass' => \Codelicia\Soql\SoqlDriver::class, 'wrapperClass' => \Codelicia\Soql\ConnectionWrapper::class, ]; /** @var \Codelicia\Soql\ConnectionWrapper $conn */ $conn = DriverManager::getConnection($connectionParams, $config);
userprovides the login, which is usually an email to access the salesforce instance.passwordprovides the corresponding password to the email provided onuser.salesforceInstancepoints to the url of the Salesforce instance.apiVersionspecify a salesforce API version to work with.consumerKeyprovides the integration consumer keyconsumerSecretprovides the integration consumer secretdriverClassshould points to\Codelicia\Soql\SoqlDriver::classwrapperClassshould points to\Codelicia\Soql\ConnectionWrapper::class
By setting up the wrapperClass, we can make use of a proper QueryBuild that allow
JOIN in the Salesforce format.
When using the doctrine bundle and the dbal is configured through yaml the options should be passed in a different way for the validation that the bundle does.
doctrine: dbal: driver: soql user: '%env(resolve:SALESFORCE_USERNAME)%' password: '%env(resolve:SALESFORCE_PASSWORD)%' driver_class: '\Codelicia\Soql\SoqlDriver' wrapper_class: '\Codelicia\Soql\ConnectionWrapper' options: salesforceInstance: '%env(resolve:SALESFORCE_ENDPOINT)%' apiVersion: v56.0 consumerKey: '%env(resolve:SALESFORCE_CLIENT_ID)%' consumerSecret: '%env(resolve:SALESFORCE_CLIENT_SECRET)%'
Using DBAL
Now that you have the connection set up, you can use Doctrine QueryBuilder to
query some data as bellow:
$id = '0062X00000vLZDVQA4'; $sql = $conn->createQueryBuilder() ->select(['Id', 'Name', 'Status__c']) ->from('Opportunity') ->where('Id = :id') ->andWhere('Name = :name') ->setParameter('name', 'Pay as you go Opportunity') ->setParameter('id', $id) ->setMaxResults(1) ->execute(); var_dump($sql->fetchAll()); // All rest api result
or use the normal Connection#query() method.
Basic Operations
Here are some examples of basic CRUD operations.
Connection#insert()
Creating an Account with the Name of John:
$connection->insert('Account', ['Name' => 'John']);
Connection#delete()
Deleting an Account with the Id = 1234:
$connection->delete('Account', ['Id' => '1234']);
Connection#update()
Update an Account with the Name of Sr. John where the Id is 1234:
$connection->update('Account', ['Name' => 'Sr. John'], ['Id' => '1234']);
Be Transactional with Composite API
As salesforce released the composite api, it gave us the ability
to simulate transactions as in a database. So, we can use the same
Doctrine DBAL api that you already know to do transactional operations
in your Salesforce instance.
$conn->beginTransaction(); $conn->insert('Account', ['Name' => 'John']); $conn->insert('Account', ['Name' => 'Elsa']); $conn->commit();
Or even, use the Connection#transactional() helper, as you prefer.
Referencing another Records
The composite api, also enables us to compose a structure data to be
changed in one single request. So we can cross reference records as it
fits our needs.
Let's see how to create an Account and a linked Contact to that Account
in a single composite request.
$conn->transactional(static function () use ($conn) { $conn->insert('Account', ['Name' => 'John'], ['referenceId' => 'account']); $conn->insert('Contact', [ 'FirstName' => 'John', 'LastName' => 'Contact', 'AccountId' => '@{account.id}' // reference `Account` by its `referenceId` ]); });
🚫 Known Limitations
As of today, we cannot consume a sObject using the queryBuilder to get all fields from
the sObject. That is because Salesforce doesn't accept SELECT * as a valid query.
The workaround that issue is to do a GET request to specific resources, then can grab all
data related to that resource.
$this->connection ->getNativeConnection() // : \GuzzleHttp\ClientInterface ->request( 'GET', sprintf('/services/data/v40.0/sobjects/Opportunity/%s', $id) ) ->getBody() ->getContents() ;
📈 Diagram
%%{init: {'sequence': { 'mirrorActors': false, 'rightAngles': true, 'messageAlign': 'center', 'actorFontSize': 20, 'actorFontWeight': 900, 'noteFontSize': 18, 'noteFontWeight': 600, 'messageFontSize': 20}}}%%
%%{init: {'theme': 'base', 'themeVariables': { 'actorBorder': '#D86613', 'activationBorderColor': '#232F3E', 'activationBkgColor': '#D86613','noteBorderColor': '#232F3E', 'signalColor': 'white', 'signalTextColor': 'gray', 'sequenceNumberColor': '#232F3E'}}}%%
sequenceDiagram
autonumber
Note left of ConnectionWrapper: Everything starts with <br/>the ConnectionWrapper.
ConnectionWrapper->>QueryBuilder: createQueryBuilder()
activate QueryBuilder
alt
QueryBuilder->>QueryBuilder: execute() <br>Calls private executeQuery()<br>method
end
QueryBuilder->>+ConnectionWrapper: executeQuery()
deactivate QueryBuilder
ConnectionWrapper->>SoqlStatement: execute()
SoqlStatement->>+\Doctrine\DBAL\Driver\Result: execute()
ConnectionWrapper->>+\Codelicia\Soql\DBAL\Result: new
\Doctrine\DBAL\Driver\Result-->>\Codelicia\Soql\DBAL\Result: pass to
\Codelicia\Soql\DBAL\Result-->>-ConnectionWrapper: returns
ConnectionWrapper->>-SoqlStatement: fetchAll()
SoqlStatement->>+\Codelicia\Soql\FetchDataUtility: fetchAll()
\Codelicia\Soql\FetchDataUtility-->>+\GuzzleHttp\ClientInterface: send()
Note right of \Codelicia\Soql\FetchDataUtility: Countable goes here?<br> before creating the Payload?
\Codelicia\Soql\FetchDataUtility->>+\Codelicia\Soql\Payload: new
\Codelicia\Soql\Payload-->>+SoqlStatement: returns
Loading
Author
- Jefersson Nathan (@malukenho)
- Alexandre Eher (@Eher)