Реализация синхронизации цен с дропшиппинг-поставщиком
Цены поставщика меняются по расписанию (новый прайс раз в неделю) или в реальном времени (биржевые товары, курсовая зависимость, динамическое ценообразование). В обоих случаях магазин не может работать с устаревшим прайсом — либо теряет маржу при росте закупочной цены, либо завышает цены при снижении, проигрывая конкурентам.
Источники цен
- REST API — эндпоинт с актуальными ценами, можно запрашивать часто
- FTP/CSV прайс-лист — файл обновляется раз в сутки или реже
- Webhook от поставщика — push при каждом изменении цены
- Парсинг сайта поставщика — крайний вариант при отсутствии API (требует обслуживания)
Хранение ценовой истории
Schema::create('dropship_price_log', function (Blueprint $table) {
$table->id();
$table->foreignId('dropship_product_id')->constrained();
$table->decimal('prev_supplier_price', 10, 2);
$table->decimal('new_supplier_price', 10, 2);
$table->decimal('prev_retail_price', 10, 2)->nullable();
$table->decimal('new_retail_price', 10, 2)->nullable();
$table->decimal('margin_percent', 5, 2)->nullable();
$table->string('source')->default('sync');
$table->timestamp('recorded_at');
$table->index(['dropship_product_id', 'recorded_at']);
});
Калькулятор розничной цены
Розничная цена рассчитывается из закупочной с применением правил маржи. Правила могут быть глобальными, на уровне поставщика, категории или конкретного товара.
class PriceCalculator
{
public function calculate(DropshipProduct $dp): float
{
$supplierPrice = $dp->supplier_price;
$marginRule = $this->resolveMarginRule($dp);
return match($marginRule->type) {
'percent' => round($supplierPrice * (1 + $marginRule->value / 100), 2),
'fixed' => round($supplierPrice + $marginRule->value, 2),
'markup_table' => $this->applyMarkupTable($supplierPrice, $marginRule->table),
};
}
private function resolveMarginRule(DropshipProduct $dp): MarginRule
{
// Приоритет: товар > категория > поставщик > глобальные настройки
return $dp->margin_rule
?? $dp->product?->category?->margin_rule
?? $dp->supplier->margin_rule
?? MarginRule::getDefault();
}
/**
* Ступенчатая наценка: дорогие товары — меньший %
* [до 1000 → +40%, 1000–5000 → +25%, 5000+ → +15%]
*/
private function applyMarkupTable(float $price, array $table): float
{
foreach ($table as $tier) {
if ($price <= $tier['max_price']) {
return round($price * (1 + $tier['percent'] / 100), 2);
}
}
// Последний уровень без ограничения максимума
$last = end($table);
return round($price * (1 + $last['percent'] / 100), 2);
}
}
Job синхронизации цен
class SyncSupplierPricesJob implements ShouldQueue
{
public $tries = 3;
public $backoff = [60, 300, 900];
public function handle(
SupplierConnectorFactory $factory,
PriceCalculator $calculator,
): void {
$connector = $factory->make($this->supplier);
$priceList = $connector->getPriceList(); // массив [sku => price]
$updatedCount = 0;
foreach ($priceList as $sku => $newSupplierPrice) {
$dp = DropshipProduct::where([
'supplier_id' => $this->supplier->id,
'supplier_sku' => $sku,
])->first();
if (!$dp) continue;
// Пропускаем, если цена не изменилась
if (abs($dp->supplier_price - $newSupplierPrice) < 0.01) continue;
$prevSupplierPrice = $dp->supplier_price;
$prevRetailPrice = $dp->product?->price;
$dp->update(['supplier_price' => $newSupplierPrice]);
// Пересчитываем розничную цену, если нет ручной фиксации
$newRetailPrice = null;
if ($dp->product && !$dp->product->price_locked) {
$newRetailPrice = $calculator->calculate($dp);
$dp->product->update(['price' => $newRetailPrice]);
}
DropshipPriceLog::create([
'dropship_product_id' => $dp->id,
'prev_supplier_price' => $prevSupplierPrice,
'new_supplier_price' => $newSupplierPrice,
'prev_retail_price' => $prevRetailPrice,
'new_retail_price' => $newRetailPrice,
'source' => 'sync',
'recorded_at' => now(),
]);
$updatedCount++;
}
Log::info('Price sync completed', [
'supplier' => $this->supplier->slug,
'updated' => $updatedCount,
]);
}
}
Защита от резких скачков цены
Иногда в прайс-листе поставщика появляются ошибочные цены (ноль, очень высокое значение, опечатка). Без защиты магазин выставит некорректную розничную цену:
class PriceSanityChecker
{
private const MAX_CHANGE_PERCENT = 50; // не обновляем, если изменение > 50%
public function isSafe(float $prevPrice, float $newPrice): bool
{
if ($newPrice <= 0) return false;
if ($prevPrice <= 0) return true; // первая цена — принимаем любую положительную
$changePercent = abs($newPrice - $prevPrice) / $prevPrice * 100;
if ($changePercent > self::MAX_CHANGE_PERCENT) {
// Логируем для ручной проверки
Log::warning('Suspicious price change detected', [
'prev' => $prevPrice,
'new' => $newPrice,
'change%' => round($changePercent, 1),
]);
return false;
}
return true;
}
}
Товары с подозрительным изменением цены помещаются в очередь ручной проверки — менеджер видит их в отдельном разделе админки.
Валюта и курсовые пересчёты
Если поставщик выставляет цены в USD или EUR, а магазин работает в RUB:
class CurrencyPriceConverter
{
public function convert(float $price, string $fromCurrency, string $toCurrency): float
{
if ($fromCurrency === $toCurrency) return $price;
$rate = Cache::remember(
"exchange_rate_{$fromCurrency}_{$toCurrency}",
3600, // кэш на 1 час
fn() => $this->fetchRate($fromCurrency, $toCurrency)
);
return round($price * $rate, 2);
}
private function fetchRate(string $from, string $to): float
{
// Центробанк РФ: cbr.ru/scripts/XML_daily.asp
// Или openexchangerates.org, fixer.io
$response = Http::get('https://api.exchangerate-api.com/v4/latest/' . $from);
return $response->json("rates.{$to}");
}
}
Расписание
// Синхронизация цен каждые 6 часов
$schedule->job(SyncAllSupplierPricesJob::class)->everySixHours()->withoutOverlapping();
Сроки
Синхронизация цен с одним поставщиком + калькулятор маржи — 3–4 рабочих дня. Ступенчатая наценка, курсовые пересчёты, защита от скачков — ещё 1–2 дня.







