From 7f0cef634ab828f94449dc40a0d3b170c829d812 Mon Sep 17 00:00:00 2001 From: Andrei Daniel Petrica Date: Wed, 3 Jul 2024 14:52:42 +0200 Subject: [PATCH 1/2] Added get getContentVersionBody and getAttachmentBody Based on code provided in issue https://github.com/omniphx/forrest/issues/179 --- src/Omniphx/Forrest/Client.php | 276 +++++++++++++++++++-------------- 1 file changed, 157 insertions(+), 119 deletions(-) diff --git a/src/Omniphx/Forrest/Client.php b/src/Omniphx/Forrest/Client.php index 754859e..1431f31 100644 --- a/src/Omniphx/Forrest/Client.php +++ b/src/Omniphx/Forrest/Client.php @@ -3,27 +3,24 @@ namespace Omniphx\Forrest; use GuzzleHttp\ClientInterface; +use GuzzleHttp\Exception\GuzzleException; use GuzzleHttp\Exception\RequestException; -use GuzzleHttp\Message\ResponseInterface; use Omniphx\Forrest\Exceptions\InvalidLoginCreditialsException; use Omniphx\Forrest\Exceptions\SalesforceException; use Omniphx\Forrest\Exceptions\TokenExpiredException; -use Omniphx\Forrest\Exceptions\MissingVersionException; - +use Omniphx\Forrest\Formatters\BaseFormatter; +use Omniphx\Forrest\Formatters\CsvFormatter; +use Omniphx\Forrest\Formatters\JSONFormatter; +use Omniphx\Forrest\Formatters\XMLFormatter; use Omniphx\Forrest\Interfaces\AuthenticationInterface; use Omniphx\Forrest\Interfaces\EncryptorInterface; use Omniphx\Forrest\Interfaces\EventInterface; +use Omniphx\Forrest\Interfaces\FormatterInterface; use Omniphx\Forrest\Interfaces\InputInterface; use Omniphx\Forrest\Interfaces\RedirectInterface; -use Omniphx\Forrest\Interfaces\FormatterInterface; use Omniphx\Forrest\Interfaces\RepositoryInterface; use Omniphx\Forrest\Interfaces\ResourceRepositoryInterface; - -use Omniphx\Forrest\Formatters\JSONFormatter; -use Omniphx\Forrest\Formatters\URLEncodedFormatter; -use Omniphx\Forrest\Formatters\XMLFormatter; -use Omniphx\Forrest\Formatters\BaseFormatter; -use Omniphx\Forrest\Formatters\CsvFormatter; +use Psr\Http\Message\ResponseInterface; /** * API resources. @@ -120,10 +117,10 @@ abstract class Client implements AuthenticationInterface */ protected $credentials; - /** @var \Omniphx\Forrest\Interfaces\RepositoryInterface */ + /** @var \Omniphx\Forrest\Interfaces\RepositoryInterface */ protected $instanceURLRepo; - /** @var \Omniphx\Forrest\Interfaces\RepositoryInterface */ + /** @var \Omniphx\Forrest\Interfaces\RepositoryInterface */ protected $refreshTokenRepo; /** @@ -162,28 +159,27 @@ public function __construct( FormatterInterface $formatter, $settings) { - $this->httpClient = $httpClient; - $this->encryptor = $encryptor; - $this->event = $event; - $this->input = $input; - $this->redirect = $redirect; - $this->instanceURLRepo = $instanceURLRepo; + $this->httpClient = $httpClient; + $this->encryptor = $encryptor; + $this->event = $event; + $this->input = $input; + $this->redirect = $redirect; + $this->instanceURLRepo = $instanceURLRepo; $this->refreshTokenRepo = $refreshTokenRepo; - $this->resourceRepo = $resourceRepo; - $this->stateRepo = $stateRepo; - $this->tokenRepo = $tokenRepo; - $this->versionRepo = $versionRepo; - $this->formatter = $formatter; - $this->settings = $settings; - $this->credentials = $settings['credentials']; + $this->resourceRepo = $resourceRepo; + $this->stateRepo = $stateRepo; + $this->tokenRepo = $tokenRepo; + $this->versionRepo = $versionRepo; + $this->formatter = $formatter; + $this->settings = $settings; + $this->credentials = $settings['credentials']; } /** * Try requesting token, if token expired try refreshing token. * - * @param string $url - * @param array $options - * + * @param string $url + * @param array $options * @return string|array */ public function request($url, $options) @@ -197,11 +193,13 @@ public function request($url, $options) $this->refresh(); $this->url = $url; + return $this->handleRequest(); } } - public function setCredentials($credentials) { + public function setCredentials($credentials) + { $this->credentials = array_replace_recursive($this->credentials, $credentials); } @@ -257,10 +255,9 @@ private function handleRequest() /** * GET method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function get($path, $requestBody = [], $options = []) @@ -271,10 +268,9 @@ public function get($path, $requestBody = [], $options = []) /** * POST method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function post($path, $requestBody = [], $options = []) @@ -285,10 +281,9 @@ public function post($path, $requestBody = [], $options = []) /** * PUT method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function put($path, $requestBody = [], $options = []) @@ -299,10 +294,9 @@ public function put($path, $requestBody = [], $options = []) /** * DELETE method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function delete($path, $requestBody = [], $options = []) @@ -313,10 +307,9 @@ public function delete($path, $requestBody = [], $options = []) /** * HEAD method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function head($path, $requestBody = [], $options = []) @@ -327,10 +320,9 @@ public function head($path, $requestBody = [], $options = []) /** * PATCH method call using any custom path. * - * @param string $path - * @param array $requestBody - * @param array $options - * + * @param string $path + * @param array $requestBody + * @param array $options * @return mixed */ public function patch($path, $requestBody = [], $options = []) @@ -341,10 +333,6 @@ public function patch($path, $requestBody = [], $options = []) /** * Prepares options and sends the request. * - * @param $path - * @param $requestBody - * @param $options - * @param $method * * @return mixed */ @@ -354,7 +342,7 @@ private function sendRequest($path, $requestBody, $options, $method) $url .= '/'.trim($path, "/\t\n\r\0\x0B"); $options['method'] = $method; - if (!empty($requestBody)) { + if (! empty($requestBody)) { $options['body'] = $requestBody; } @@ -367,8 +355,7 @@ private function sendRequest($path, $requestBody, $options, $method) * Formats: json, xml * Methods: get. * - * @param array $options - * + * @param array $options * @return array $versions */ public function versions($options = []) @@ -387,8 +374,7 @@ public function versions($options = []) * Formats: json, xml * Methods: get. * - * @param array $options - * + * @param array $options * @return array $resources */ public function resources($options = []) @@ -403,7 +389,6 @@ public function resources($options = []) * Returns information about the logged-in user. * * @param array - * * @return array $identity */ public function identity($options = []) @@ -425,8 +410,7 @@ public function identity($options = []) * Available for API version 29.0 and later. * Returns limits for daily API calls, Data storage, etc. * - * @param array $options - * + * @param array $options * @return array $limits */ public function limits($options = []) @@ -442,16 +426,15 @@ public function limits($options = []) /** * Describes all global objects available in the organization. * - * @param string $object_name - * @param array $options - * + * @param string $object_name + * @param array $options * @return array */ public function describe($object_name = null, $options = []) { $url = sprintf('%s/sobjects', $this->getBaseUrl()); - if (!empty($object_name)) { + if (! empty($object_name)) { $url .= sprintf('/%s/describe', $object_name); } @@ -461,9 +444,8 @@ public function describe($object_name = null, $options = []) /** * Executes a specified SOQL query. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array $queryResults */ public function query($query, $options = []) @@ -481,9 +463,7 @@ public function query($query, $options = []) /** * Calls next query. * - * @param $nextUrl - * @param array $options - * + * @param array $options * @return mixed */ public function next($nextUrl, $options = []) @@ -500,9 +480,8 @@ public function next($nextUrl, $options = []) * Details how Salesforce will process your query. * Available for API verison 30.0 or later. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array $queryExplain */ public function queryExplain($query, $options = []) @@ -522,9 +501,8 @@ public function queryExplain($query, $options = []) * been deleted. * Available for API version 29.0 or later. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array $queryResults */ public function queryAll($query, $options = []) @@ -542,9 +520,8 @@ public function queryAll($query, $options = []) /** * Executes the specified SOSL query. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array */ public function search($query, $options = []) @@ -566,8 +543,7 @@ public function search($query, $options = []) * search results accordingly. Objects used most frequently appear * at the top of the list. * - * @param array $options - * + * @param array $options * @return array */ public function scopeOrder($options = []) @@ -584,9 +560,8 @@ public function scopeOrder($options = []) /** * Returns search result layout information for the objects in the query string. * - * @param array $objectList - * @param array $options - * + * @param array $objectList + * @param array $options * @return array */ public function searchLayouts($objectList, $options = []) @@ -607,9 +582,8 @@ public function searchLayouts($objectList, $options = []) * relevant articles, before the user performs a search. * Available for API version 30.0 or later. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array */ public function suggestedArticles($query, $options = []) @@ -620,7 +594,7 @@ public function suggestedArticles($query, $options = []) $url .= urlencode($query); $parameters = [ - 'language' => $this->settings['language'], + 'language' => $this->settings['language'], 'publishStatus' => 'Online', ]; @@ -641,9 +615,8 @@ public function suggestedArticles($query, $options = []) * * Tested this and can't get it to work. I think the request is set up correctly. * - * @param string $query - * @param array $options - * + * @param string $query + * @param array $options * @return array */ public function suggestedQueries($query, $options = []) @@ -668,9 +641,8 @@ public function suggestedQueries($query, $options = []) /** * Request to a custom Apex REST endpoint. * - * @param string $customURI - * @param array $options - * + * @param string $customURI + * @param array $options * @return mixed */ public function custom($customURI, $options = []) @@ -725,9 +697,8 @@ public function getToken() * methods that can be called or refence them by calling the * Session::get('resources') method. * - * @param string $name - * @param array $arguments - * + * @param string $name + * @param array $arguments * @return string|array */ public function __call($name, $arguments) @@ -741,14 +712,20 @@ public function __call($name, $arguments) return $this->request($url, $options); } - private function appendURL($arguments) { - if (!isset($arguments[0])) return ''; - if (!is_string($arguments[0])) return ''; + private function appendURL($arguments) + { + if (! isset($arguments[0])) { + return ''; + } + if (! is_string($arguments[0])) { + return ''; + } return "/$arguments[0]"; } - private function setOptions($arguments) { + private function setOptions($arguments) + { $options = []; foreach ($arguments as $argument) { @@ -758,8 +735,11 @@ private function setOptions($arguments) { return $options; } - private function setArgument($argument, &$options) { - if (!is_array($argument)) return; + private function setArgument($argument, &$options) + { + if (! is_array($argument)) { + return; + } foreach ($argument as $key => $value) { $options[$key] = $value; } @@ -806,15 +786,15 @@ protected function storeVersion() /** * Overrides the default formatter set during register. * - * @param string $formatter - Name of the formatter to use + * @param string $formatter - Name of the formatter to use */ protected function setFormatter($formatter) { if ($formatter === 'json' && strpos(get_class($this->formatter), 'JSONFormatter') === false) { $this->formatter = new JSONFormatter($this->tokenRepo, $this->settings); - } else if ($formatter === 'xml' && strpos(get_class($this->formatter), 'XMLFormatter') === false) { + } elseif ($formatter === 'xml' && strpos(get_class($this->formatter), 'XMLFormatter') === false) { $this->formatter = new XMLFormatter($this->tokenRepo, $this->settings); - } else if ($formatter === 'none' && strpos(get_class($this->formatter), 'BaseFormatter') === false) { + } elseif ($formatter === 'none' && strpos(get_class($this->formatter), 'BaseFormatter') === false) { $this->formatter = new BaseFormatter($this->tokenRepo, $this->settings); } elseif ($formatter === 'csv' && strpos(get_class($this->formatter), 'CsvFormatter') === false) { $this->formatter = new CsvFormatter($this->tokenRepo, $this->settings); @@ -824,16 +804,20 @@ protected function setFormatter($formatter) private function storeConfiguredVersion($versions) { $configVersion = $this->settings['version']; - if (empty($configVersion)) return; + if (empty($configVersion)) { + return; + } - foreach($versions as $version) { + foreach ($versions as $version) { $this->determineIfConfiguredVersionExists($version, $configVersion); } } private function determineIfConfiguredVersionExists($version, $configVersion) { - if ($version['version'] !== $configVersion) return; + if ($version['version'] !== $configVersion) { + return; + } $this->versionRepo->put($version); } @@ -858,7 +842,9 @@ protected function storeResources() protected function handleAuthenticationErrors(array $response) { - if (!isset($response['error'])) return; + if (! isset($response['error'])) { + return; + } throw new InvalidLoginCreditialsException($response['error_description']); } @@ -866,24 +852,76 @@ protected function handleAuthenticationErrors(array $response) /** * Method will elaborate on RequestException. * - * @param RequestException $ex * * @throws SalesforceException * @throws TokenExpiredException */ private function assignExceptions(RequestException $ex) { - if ($ex->hasResponse() && 401 == $ex->getResponse()->getStatusCode()) { + if ($ex->hasResponse() && $ex->getResponse()->getStatusCode() == 401) { throw new TokenExpiredException('Salesforce token has expired', $ex); - } elseif ($ex->hasResponse() && 403 == $ex->getResponse()->getStatusCode() && 'Bad_OAuth_Token' == $ex->getResponse()->getBody()->getContents()) { + } elseif ($ex->hasResponse() && $ex->getResponse()->getStatusCode() == 403 && $ex->getResponse()->getBody()->getContents() == 'Bad_OAuth_Token') { throw new TokenExpiredException('Salesforce token has expired', $ex); } elseif ($ex->hasResponse()) { $error = json_decode($ex->getResponse()->getBody()->getContents(), true); $ex->getResponse()->getBody()->rewind(); - $jsonError = json_encode($error,JSON_PRETTY_PRINT); + $jsonError = json_encode($error, JSON_PRETTY_PRINT); throw new SalesforceException($jsonError, $ex); } else { throw new SalesforceException(sprintf('Invalid request: %s', $ex->getMessage()), $ex); } } + + /** + * @param string $id Attachment.Id + * + * @throws GuzzleException + */ + public function getAttachmentBody(string $id): ResponseInterface + { + $url = $this->getBaseUrl().sprintf('/sobjects/Attachment/%s/body', $id); + + $parameters = [ + 'headers' => $this->formatter->setHeaders(), + ]; + + $response = $this->httpClient->request('get', $url, $parameters); + + $this->event->fire('forrest.response', json_encode([ + 'url' => $url, + 'method' => 'get', + 'status_code' => $response->getStatusCode(), + 'body_size' => (string) $response->getBody()->getSize(), + ])); + + return $response; + } + + /** + * @param string $id ContentVersion.Id + * + * @throws GuzzleException + */ + public function getContentVersionBody(string $id): ResponseInterface + { + + $url = $this->getBaseUrl().sprintf('/sobjects/ContentVersion/%s/VersionData', $id); + + $parameters = [ + 'headers' => $this->formatter->setHeaders(), + ]; + + $response = $this->httpClient->request('get', $url, $parameters); + + $this->event->fire( + 'forrest.response', + json_encode([ + 'url' => $url, + 'method' => 'get', + 'status_code' => $response->getStatusCode(), + 'body_size' => (string) $response->getBody()->getSize(), + ])); + + return $response; + } } From 79b9901c941e10ddbead517d7a0aa32634c1d810 Mon Sep 17 00:00:00 2001 From: Andrei Daniel Petrica Date: Wed, 3 Jul 2024 15:24:10 +0200 Subject: [PATCH 2/2] Readme and facade update --- README.md | 10 ++++++++++ .../Forrest/Providers/Laravel/Facades/Forrest.php | 3 +++ 2 files changed, 13 insertions(+) diff --git a/README.md b/README.md index b5db671..daeba9e 100644 --- a/README.md +++ b/README.md @@ -571,6 +571,16 @@ Forrest::put('/services/data/v20.0/endpoint', ['my'=>'param']); Forrest::patch('/services/data/v20.0/endpoint', ['my'=>'param']); Forrest::delete('/services/data/v20.0/endpoint'); ``` +### Get file body from ContentVersion and Attachment +You can use the Forrest::getContentVersionBody() and Forrest::getAttachmentBody() to retrieve the content of the +uploaded files. They return a streamed response, so it may be a bit cumbersome to use if now used to streams. +Bellow you can find an example to retrieve the content of a uploaded content version. + +```php +# example +$data = Forrest::getContentVersionBody($version->Id); +$content = $data->getBody()->getContents(); +``` ### Raw response output diff --git a/src/Omniphx/Forrest/Providers/Laravel/Facades/Forrest.php b/src/Omniphx/Forrest/Providers/Laravel/Facades/Forrest.php index fdf6644..ee7ac33 100644 --- a/src/Omniphx/Forrest/Providers/Laravel/Facades/Forrest.php +++ b/src/Omniphx/Forrest/Providers/Laravel/Facades/Forrest.php @@ -3,6 +3,7 @@ namespace Omniphx\Forrest\Providers\Laravel\Facades; use Illuminate\Support\Facades\Facade; +use Psr\Http\Message\ResponseInterface; /** * @method static \Illuminate\Http\RedirectResponse|void authenticate() @@ -54,6 +55,8 @@ * @method static string|array sobjects(string $resource = "", array $options = []) * @method static string|array actions(string $resource, array $options = []) * @method static string|array support(string $resource, array $options = []) + * @method static ResponseInterface getAttachmentBody(string $id) + * @method static ResponseInterface getContentVersionBody(string $id) * @method static \Omniphx\Forrest\Interfaces\RedirectInterface callback() */ class Forrest extends Facade