mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-14 23:22:52 +01:00
214 lines
5.8 KiB
PHP
214 lines
5.8 KiB
PHP
<?php
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\Services\Quickbooks;
|
|
|
|
use App\DataMapper\QuickbooksSettings;
|
|
use Carbon\Carbon;
|
|
use App\Models\Company;
|
|
use QuickBooksOnline\API\DataService\DataService;
|
|
use QuickBooksOnline\API\Core\OAuth\OAuth2\OAuth2AccessToken;
|
|
|
|
class SdkWrapper
|
|
{
|
|
public const MAXRESULTS = 10000;
|
|
|
|
private $entities = ['Customer','Invoice','Item','SalesReceipt','Vendor'];
|
|
|
|
private OAuth2AccessToken $token;
|
|
|
|
public function __construct(public DataService $sdk, private Company $company)
|
|
{
|
|
$this->init();
|
|
}
|
|
|
|
private function init(): self
|
|
{
|
|
|
|
isset($this->company->quickbooks->accessTokenKey) ? $this->setNinjaAccessToken($this->company->quickbooks) : null;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
public function getAuthorizationUrl(): string
|
|
{
|
|
return $this->sdk->getOAuth2LoginHelper()->getAuthorizationCodeURL();
|
|
}
|
|
|
|
public function getState(): string
|
|
{
|
|
return $this->sdk->getOAuth2LoginHelper()->getState();
|
|
}
|
|
|
|
public function getRefreshToken(): string
|
|
{
|
|
return $this->accessToken()->getRefreshToken();
|
|
}
|
|
|
|
public function company()
|
|
{
|
|
return $this->sdk->getCompanyInfo();
|
|
}
|
|
/*
|
|
accessTokenKey
|
|
tokenType
|
|
refresh_token
|
|
accessTokenExpiresAt
|
|
refreshTokenExpiresAt
|
|
accessTokenValidationPeriod
|
|
refreshTokenValidationPeriod
|
|
clientID
|
|
clientSecret
|
|
realmID
|
|
baseURL
|
|
*/
|
|
public function accessTokenFromCode(string $code, string $realm): OAuth2AccessToken
|
|
{
|
|
$token = $this->sdk->getOAuth2LoginHelper()->exchangeAuthorizationCodeForToken($code, $realm);
|
|
|
|
$this->setAccessToken($token);
|
|
|
|
return $this->accessToken();
|
|
}
|
|
|
|
/**
|
|
* Set Stored NinjaAccessToken
|
|
*
|
|
* @param QuickbooksSettings $token_object
|
|
* @return self
|
|
*/
|
|
public function setNinjaAccessToken(QuickbooksSettings $token_object): self
|
|
{
|
|
$token = new OAuth2AccessToken(
|
|
config('services.quickbooks.client_id'),
|
|
config('services.quickbooks.client_secret'),
|
|
$token_object->accessTokenKey,
|
|
$token_object->refresh_token,
|
|
3600,
|
|
8726400
|
|
);
|
|
|
|
$token->setAccessTokenExpiresAt($token_object->accessTokenExpiresAt); //@phpstan-ignore-line
|
|
$token->setRefreshTokenExpiresAt($token_object->refreshTokenExpiresAt); //@phpstan-ignore-line
|
|
$token->setAccessTokenValidationPeriodInSeconds(3600);
|
|
$token->setRefreshTokenValidationPeriodInSeconds(8726400);
|
|
|
|
$this->setAccessToken($token);
|
|
|
|
if($token_object->accessTokenExpiresAt < time()){
|
|
$new_token = $this->sdk->getOAuth2LoginHelper()->refreshToken();
|
|
|
|
$this->setAccessToken($new_token);
|
|
$this->saveOAuthToken($this->accessToken());
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* SetsAccessToken
|
|
*
|
|
* @param OAuth2AccessToken $token
|
|
* @return self
|
|
*/
|
|
public function setAccessToken(OAuth2AccessToken $token): self
|
|
{
|
|
$this->token = $token;
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function accessToken(): OAuth2AccessToken
|
|
{
|
|
return $this->token;
|
|
}
|
|
|
|
public function saveOAuthToken(OAuth2AccessToken $token): void
|
|
{
|
|
$obj = $this->company->quickbooks ?? new QuickbooksSettings();
|
|
$obj->accessTokenKey = $token->getAccessToken();
|
|
$obj->refresh_token = $token->getRefreshToken();
|
|
$obj->accessTokenExpiresAt = Carbon::createFromFormat('Y/m/d H:i:s', $token->getAccessTokenExpiresAt())->timestamp; //@phpstan-ignore-line - QB phpdoc wrong types!!
|
|
$obj->refreshTokenExpiresAt = Carbon::createFromFormat('Y/m/d H:i:s', $token->getRefreshTokenExpiresAt())->timestamp; //@phpstan-ignore-line - QB phpdoc wrong types!!
|
|
|
|
$obj->realmID = $token->getRealmID();
|
|
$obj->baseURL = $token->getBaseURL();
|
|
|
|
$this->company->quickbooks = $obj;
|
|
$this->company->save();
|
|
}
|
|
|
|
|
|
/// Data Access ///
|
|
|
|
public function totalRecords(string $entity): int
|
|
{
|
|
return (int)$this->sdk->Query("select count(*) from $entity");
|
|
}
|
|
|
|
private function queryData(string $query, int $start = 1, $limit = 1000): array
|
|
{
|
|
return (array) $this->sdk->Query($query, $start, $limit);
|
|
}
|
|
|
|
public function fetchById(string $entity, $id)
|
|
{
|
|
return $this->sdk->FindById($entity, $id);
|
|
}
|
|
|
|
public function fetchRecords(string $entity, int $max = 100000): array
|
|
{
|
|
|
|
if(!in_array($entity, $this->entities)) {
|
|
return [];
|
|
}
|
|
|
|
$records = [];
|
|
$start = 0;
|
|
$limit = 1000;
|
|
try {
|
|
|
|
$total = $this->totalRecords($entity);
|
|
$total = min($max, $total);
|
|
|
|
nlog("total = {$total}");
|
|
|
|
// Step 3 & 4: Get chunks of records until the total required records are retrieved
|
|
do {
|
|
$limit = min(self::MAXRESULTS, $total - $start);
|
|
|
|
nlog("limit = {$limit}");
|
|
|
|
$recordsChunk = $this->queryData("select * from $entity", $start, $limit);
|
|
if(empty($recordsChunk)) {
|
|
break;
|
|
}
|
|
|
|
$records = array_merge($records, $recordsChunk);
|
|
$start += $limit;
|
|
nlog("start = {$start}");
|
|
|
|
} while ($start < $total);
|
|
if(empty($records)) {
|
|
throw new \Exception("No records retrieved!");
|
|
}
|
|
|
|
} catch (\Throwable $th) {
|
|
nlog("Fetch Quickbooks API Error: {$th->getMessage()}");
|
|
}
|
|
|
|
// nlog($records);
|
|
|
|
return $records;
|
|
}
|
|
}
|