Интеграция дропшиппинг-поставщиков с интернет-магазином
Интеграция поставщика — техническая задача, сложность которой определяется не количеством товаров, а форматом и качеством данных на стороне поставщика. REST API с документацией — лучший сценарий. Прайс в Excel без артикулов и с кириллицей в заголовках — худший. Встречаются оба.
Типы интеграций
REST API — поставщик предоставляет endpoint'ы для получения каталога, остатков, цен и приёма заказов. Наиболее удобный формат. Требует ключа API или OAuth-авторизации.
SOAP/XML-RPC — устаревший, но всё ещё распространённый формат у крупных дистрибьюторов и производителей. Требует парсинга WSDL и генерации клиентского кода.
FTP/SFTP + CSV/XML — поставщик выкладывает файл на сервер по расписанию. Магазин забирает его и обрабатывает. Нет возможности проверить остаток в реальном времени.
Email с прайс-листом — крайний случай. Применяется парсер вложений + OCR для PDF.
EDI (EDIFACT/X12) — используется крупными FMCG и фармацевтическими дистрибьюторами.
Фабрика коннекторов
class SupplierConnectorFactory
{
public static function make(Supplier $supplier): SupplierConnectorInterface
{
return match($supplier->integration_type) {
'rest_api' => new RestApiConnector($supplier, app(HttpClient::class)),
'soap' => new SoapConnector($supplier),
'ftp_csv' => new FtpCsvConnector($supplier, app(SftpFilesystem::class)),
'ftp_xml' => new FtpXmlConnector($supplier, app(SftpFilesystem::class)),
default => throw new UnsupportedIntegrationTypeException($supplier->integration_type),
};
}
}
Коннектор FTP + CSV
Часть поставщиков выгружает прайс на FTP раз в сутки. Коннектор забирает файл, парсит и нормализует данные:
class FtpCsvConnector implements SupplierConnectorInterface
{
public function getProducts(int $page = 1, int $perPage = 100): array
{
$localPath = $this->downloadFile();
$products = [];
$handle = fopen($localPath, 'r');
$headers = fgetcsv($handle, 0, ';');
$headers = array_map('trim', $headers); // убираем BOM и пробелы
// Маппинг заголовков (поставщики называют поля по-разному)
$mapping = $this->resolveHeaderMapping($headers);
while (($row = fgetcsv($handle, 0, ';')) !== false) {
$normalized = $this->normalizeRow(
array_combine($headers, $row),
$mapping
);
if ($normalized) {
$products[] = $normalized;
}
}
fclose($handle);
@unlink($localPath);
return array_slice($products, ($page - 1) * $perPage, $perPage);
}
private function resolveHeaderMapping(array $headers): array
{
// Разные поставщики используют разные названия одних и тех же полей
$aliases = [
'sku' => ['артикул', 'sku', 'код', 'article', 'item_no'],
'name' => ['наименование', 'название', 'name', 'title', 'товар'],
'price' => ['цена', 'price', 'стоимость', 'цена_розница'],
'stock' => ['остаток', 'количество', 'stock', 'qty', 'available'],
];
$mapping = [];
foreach ($headers as $header) {
$lower = mb_strtolower(trim($header));
foreach ($aliases as $field => $list) {
if (in_array($lower, $list)) {
$mapping[$field] = $header;
break;
}
}
}
return $mapping;
}
private function downloadFile(): string
{
$remotePath = $this->supplier->credentials['ftp_path'];
$localPath = sys_get_temp_dir() . '/' . uniqid('supplier_') . '.csv';
$this->sftp->download($remotePath, $localPath);
return $localPath;
}
}
Коннектор SOAP
class SoapConnector implements SupplierConnectorInterface
{
private \SoapClient $client;
public function __construct(private Supplier $supplier)
{
$this->client = new \SoapClient(
$supplier->credentials['wsdl_url'],
['login' => $supplier->credentials['login'],
'password' => $supplier->credentials['password'],
'cache_wsdl' => WSDL_CACHE_DISK,
'trace' => false,
]
);
}
public function getProducts(int $page = 1, int $perPage = 100): array
{
$result = $this->client->GetProductList([
'SessionID' => $this->getSession(),
'PageNum' => $page,
'PageSize' => $perPage,
]);
return collect($result->ProductList->Product ?? [])
->map(fn($item) => new SupplierProductDTO(
sku: $item->Article,
name: $item->Name,
price: (float) $item->Price,
stock: (int) $item->Qty,
))
->toArray();
}
}
Нормализация данных поставщика
Данные от разных поставщиков неизбежно расходятся по структуре. Нормализация выполняется перед сохранением в dropship_products:
class SupplierProductNormalizer
{
public function normalize(array $raw, Supplier $supplier): ?SupplierProductDTO
{
// Очищаем артикул от спецсимволов
$sku = preg_replace('/[^\w\-]/', '', $raw['sku'] ?? '');
if (!$sku) return null;
// Нормализуем цену: убираем пробелы, заменяем запятую на точку
$price = (float) str_replace([' ', ','], ['', '.'], $raw['price'] ?? '0');
if ($price <= 0) return null;
// Нормализуем остаток: "в наличии" → 999, "нет" → 0
$stock = $this->parseStock($raw['stock'] ?? '0');
return new SupplierProductDTO(
sku: $sku,
name: mb_convert_encoding(trim($raw['name'] ?? ''), 'UTF-8', 'auto'),
price: $price,
stock: $stock,
);
}
private function parseStock(mixed $value): int
{
if (is_numeric($value)) return (int) $value;
$lower = mb_strtolower((string) $value);
return match(true) {
str_contains($lower, 'наличи') => 999,
str_contains($lower, 'нет') => 0,
str_contains($lower, 'ожида') => 0,
default => 0,
};
}
}
Обработка ошибок соединения
Поставщики бывают ненадёжными: API ложится на техобслуживание, FTP меняет структуру директорий, CSV приходит с другой кодировкой. Все коннекторы оборачиваются в Retry-политику через Laravel Queue с exponential backoff:
class SyncSupplierJob implements ShouldQueue
{
public $tries = 3;
public $backoff = [60, 300, 900]; // 1 мин, 5 мин, 15 мин
public function failed(Throwable $e): void
{
Notification::route('mail', config('suppliers.admin_email'))
->notify(new SupplierSyncFailedNotification($this->supplier, $e));
}
}
Сроки интеграции
| Тип интеграции | Срок |
|---|---|
| REST API с документацией | 2–3 дня |
| SOAP с WSDL | 3–4 дня |
| FTP + CSV (стандартный формат) | 2–3 дня |
| FTP + CSV (нестандартный формат) | 3–5 дней |
| Несколько поставщиков (каждый последующий) | 1–3 дня |







