- Introduced new endpoints for creating and modifying articles (`art_add`, `art_mod`) and tiers (`third_add`, `third_mod`), allowing users to manage these entities effectively. - Updated the Articles and Tiers pages to include forms for adding and modifying records, complete with parameter tables for clear guidance on required inputs. - Enhanced the API documentation to include detailed descriptions, examples, and metadata for the new endpoints, improving usability and understanding for developers. - Created a new rule for writing conventions with French accents to ensure consistency across the project. - Updated existing documentation to reflect structural changes and added a summary table for CRUD operations. - Added tests to verify the functionality of the new features and ensure robust error handling.
268 lines
7.8 KiB
PHP
268 lines
7.8 KiB
PHP
<?php
|
|
|
|
namespace App\Services;
|
|
|
|
use App\Exceptions\LogisticsApiException;
|
|
use Illuminate\Http\Client\ConnectionException;
|
|
use Illuminate\Http\Client\Response;
|
|
use Illuminate\Support\Facades\DB;
|
|
use Illuminate\Support\Facades\Http;
|
|
|
|
class LogisticsService
|
|
{
|
|
private string $baseUrl;
|
|
|
|
private string $apiKey;
|
|
|
|
private string $folder;
|
|
|
|
private int $timeout;
|
|
|
|
private int $connectTimeout;
|
|
|
|
private int $retryTimes;
|
|
|
|
private int $retrySleepMs;
|
|
|
|
public function __construct()
|
|
{
|
|
$this->baseUrl = rtrim(config('logistics.base_url'), '/');
|
|
$this->apiKey = config('logistics.api_key');
|
|
$this->folder = config('logistics.folder');
|
|
$this->timeout = config('logistics.timeout', 30);
|
|
$this->connectTimeout = config('logistics.connect_timeout', 10);
|
|
$this->retryTimes = config('logistics.retry.times', 3);
|
|
$this->retrySleepMs = config('logistics.retry.sleep_ms', 500);
|
|
}
|
|
|
|
public function tablesList(): array
|
|
{
|
|
return $this->post('tables_list');
|
|
}
|
|
|
|
public function columnList(string $table): array
|
|
{
|
|
return $this->post("column_list/{$table}");
|
|
}
|
|
|
|
/**
|
|
* @param array{select?: string, results?: int, search?: string, barcode?: string} $params
|
|
*/
|
|
public function artList(array $params = []): array
|
|
{
|
|
return $this->post('art_list', $params);
|
|
}
|
|
|
|
public function artGetStock(string $artId): array
|
|
{
|
|
return $this->post('art_getstk', ['ARTID' => $artId]);
|
|
}
|
|
|
|
/**
|
|
* @param array{select?: string, results?: int, TYPE: string} $params
|
|
*/
|
|
public function jnlList(array $params = []): array
|
|
{
|
|
return $this->post('jnl_list', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{select?: string, thirdid?: string} $params
|
|
*/
|
|
public function documentList(array $params = []): array
|
|
{
|
|
return $this->post('document_list', $params);
|
|
}
|
|
|
|
public function documentDetail(string $jnl, string $number): array
|
|
{
|
|
return $this->post('document_detail', [
|
|
'jnl' => $jnl,
|
|
'number' => $number,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param array{ThirdId: string, Date: string, Artid: array, Qty: array, Saleprice: array, JNL: string, Discount?: array, Vatid?: array, Vatpc?: array, Attachments?: array} $params
|
|
*/
|
|
public function documentAdd(array $params): array
|
|
{
|
|
return $this->post('document_add', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{number: string, Thirdid?: string, Artid?: array, Qty?: array, Saleprice?: array, JNL: string, Attachments?: array} $params
|
|
*/
|
|
public function documentMod(array $params): array
|
|
{
|
|
return $this->post('document_mod', $params);
|
|
}
|
|
|
|
public function documentGetStatusList(string $jnl): array
|
|
{
|
|
return $this->post('Document_GetStatusList', ['jnl' => $jnl]);
|
|
}
|
|
|
|
/**
|
|
* @param array{ARTID: string, QTY: string, JNL: string, THIRDID: string, DATE: string} $params
|
|
*/
|
|
public function documentGetUnitPriceAndVat(array $params): array
|
|
{
|
|
return $this->post('Document_GetUnitPriceAndVat', $params);
|
|
}
|
|
|
|
public function documentGetDueDate(string $payDelay, string $date): array
|
|
{
|
|
return $this->post('Document_GetDueDate', [
|
|
'paydelay' => $payDelay,
|
|
'date' => $date,
|
|
]);
|
|
}
|
|
|
|
public function documentGetAttachListThumbnail(string $jnl, string $number): array
|
|
{
|
|
return $this->post('Document_GetAttachListThumbnail', [
|
|
'JNL' => $jnl,
|
|
'NUMBER' => $number,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param array{select?: string, results?: int, search: string} $params
|
|
*/
|
|
public function thirdList(array $params = []): array
|
|
{
|
|
return $this->post('third_list', $params);
|
|
}
|
|
|
|
public function thirdGetArtHistory(string $thirdId): array
|
|
{
|
|
return $this->post('third_GetArtHistory', ['thirdid' => $thirdId]);
|
|
}
|
|
|
|
public function getSerialNumber(): array
|
|
{
|
|
return $this->post('getserialnumber');
|
|
}
|
|
|
|
/**
|
|
* @param array{code: string} $params
|
|
*/
|
|
public function codesList(array $params): array
|
|
{
|
|
return $this->post('codes_list', $params);
|
|
}
|
|
|
|
public function documentGetPdf(string $jnl, string $number, string $layout): array
|
|
{
|
|
return $this->post('Document_GetPDF', [
|
|
'JNL' => $jnl,
|
|
'NUMBER' => $number,
|
|
'LAYOUT' => $layout,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param array{ARTID: string, NAME1?: string, SALEPRICE?: string, ...} $params
|
|
*/
|
|
public function artAdd(array $params): array
|
|
{
|
|
return $this->post('art_add', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{ARTID: string, NAME1?: string, SALEPRICE?: string, ...} $params
|
|
*/
|
|
public function artMod(array $params): array
|
|
{
|
|
return $this->post('art_mod', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{NAME?: string, VAT?: string, EMAIL?: string, ...} $params
|
|
*/
|
|
public function thirdAdd(array $params): array
|
|
{
|
|
return $this->post('third_add', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{CUSTID: string, NAME?: string, VAT?: string, EMAIL?: string, ...} $params
|
|
*/
|
|
public function thirdMod(array $params): array
|
|
{
|
|
return $this->post('third_mod', $params);
|
|
}
|
|
|
|
/**
|
|
* @param array{ARTID: string, STKID: string, QTY: string, TOCHECK?: string, TOCHECKDETAIL?: string, MODE?: string} $params
|
|
*/
|
|
public function customGeninvUpdatestock(array $params): array
|
|
{
|
|
return $this->post('custom_geninv_updatestock', $params);
|
|
}
|
|
|
|
/**
|
|
* @return array{data: mixed, metadata: array, error: mixed}
|
|
*
|
|
* @throws LogisticsApiException
|
|
*/
|
|
private function post(string $endpoint, array $params = []): array
|
|
{
|
|
$url = "{$this->baseUrl}/{$this->folder}/{$endpoint}";
|
|
|
|
try {
|
|
$response = Http::withHeaders([
|
|
'X-API-KEY' => $this->apiKey,
|
|
])
|
|
->timeout($this->timeout)
|
|
->connectTimeout($this->connectTimeout)
|
|
->retry($this->retryTimes, $this->retrySleepMs, fn (\Exception $e) => $e instanceof ConnectionException)
|
|
->post($url, $params);
|
|
|
|
$this->logRequest($endpoint, $params, $response);
|
|
|
|
return $response->json() ?? [
|
|
'data' => null,
|
|
'metadata' => ['rowcount' => 0, 'issuccess' => false],
|
|
'error' => 'Empty response from API',
|
|
];
|
|
} catch (ConnectionException $e) {
|
|
$this->logFailedRequest($endpoint, $params, $e);
|
|
|
|
throw LogisticsApiException::connectionTimeout($endpoint, $params, $e);
|
|
} catch (\Throwable $e) {
|
|
$this->logFailedRequest($endpoint, $params, $e);
|
|
|
|
throw LogisticsApiException::requestFailed($endpoint, $params, $e);
|
|
}
|
|
}
|
|
|
|
private function logRequest(string $endpoint, array $params, Response $response): void
|
|
{
|
|
$body = $response->body();
|
|
$isValidJson = json_decode($body) !== null || $body === 'null';
|
|
|
|
DB::table('api_request_logs')->insert([
|
|
'endpoint' => $endpoint,
|
|
'parameters' => json_encode($params),
|
|
'response_status' => $response->status(),
|
|
'response_data' => $isValidJson ? $body : json_encode(['raw' => $body]),
|
|
'created_at' => now(),
|
|
'updated_at' => now(),
|
|
]);
|
|
}
|
|
|
|
private function logFailedRequest(string $endpoint, array $params, \Throwable $exception): void
|
|
{
|
|
DB::table('api_request_logs')->insert([
|
|
'endpoint' => $endpoint,
|
|
'parameters' => json_encode($params),
|
|
'response_status' => 0,
|
|
'response_data' => json_encode(['error' => $exception->getMessage()]),
|
|
'created_at' => now(),
|
|
'updated_at' => now(),
|
|
]);
|
|
}
|
|
}
|