Интеграция дропшиппинг-поставщиков с интернет-магазином

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Интеграция дропшиппинг-поставщиков с интернет-магазином
Сложная
~5 рабочих дней
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Интеграция дропшиппинг-поставщиков с интернет-магазином

Интеграция поставщика — техническая задача, сложность которой определяется не количеством товаров, а форматом и качеством данных на стороне поставщика. 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 дня