Реализация региональных складов и доставки на сайте

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

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

Информационные сайты или веб-приложения
Сайты визитки, 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

Реализация региональных складов и доставки на сайте

Несколько складов в разных городах — это ускорение доставки и снижение её стоимости для региональных покупателей. Но без правильной логики выбора склада выигрыш превращается в проблемы: товар резервируется на дальнем складе, хотя рядом есть остатки; расчёт стоимости доставки ведётся от неправильного города; остатки показываются неагрегированно.

Задачи системы

  • Хранить остатки по каждому складу отдельно
  • При показе товара выводить суммарное наличие и ближайший склад к покупателю
  • Рассчитывать доставку от нужного склада, а не одного центрального
  • При оформлении заказа резервировать товар на конкретном складе
  • Поддерживать частичное выполнение заказа с разных складов

Схема данных

CREATE TABLE warehouses (
    id              BIGSERIAL PRIMARY KEY,
    name            VARCHAR(255) NOT NULL,
    city            VARCHAR(255) NOT NULL,
    address         TEXT,
    lat             NUMERIC(10,7),
    lng             NUMERIC(10,7),
    country         CHAR(2) DEFAULT 'RU',
    region_code     VARCHAR(20),           -- код региона ФИАС
    is_active       BOOLEAN DEFAULT TRUE,
    priority        SMALLINT DEFAULT 0    -- выше = предпочтительнее при равных условиях
);

CREATE TABLE warehouse_stocks (
    id              BIGSERIAL PRIMARY KEY,
    product_id      BIGINT REFERENCES products(id),
    warehouse_id    BIGINT REFERENCES warehouses(id),
    quantity        INT NOT NULL DEFAULT 0,
    reserved        INT NOT NULL DEFAULT 0,
    available       INT GENERATED ALWAYS AS (GREATEST(quantity - reserved, 0)) STORED,
    updated_at      TIMESTAMP DEFAULT NOW(),
    UNIQUE(product_id, warehouse_id)
);

CREATE INDEX idx_wstocks_product ON warehouse_stocks(product_id);
CREATE INDEX idx_wstocks_available ON warehouse_stocks(product_id, available) WHERE available > 0;

-- Резервирование при создании заказа
CREATE TABLE stock_reservations (
    id              BIGSERIAL PRIMARY KEY,
    order_id        BIGINT REFERENCES orders(id),
    order_item_id   BIGINT,
    warehouse_id    BIGINT REFERENCES warehouses(id),
    product_id      BIGINT REFERENCES products(id),
    quantity        INT NOT NULL,
    status          VARCHAR(20) DEFAULT 'reserved',  -- 'reserved', 'shipped', 'cancelled'
    reserved_at     TIMESTAMP DEFAULT NOW(),
    expires_at      TIMESTAMP                         -- автоотмена через 30 мин
);

Выбор склада для покупателя

class WarehouseSelector
{
    public function selectForDelivery(
        int    $productId,
        int    $quantity,
        string $destinationCity,
    ): ?WarehouseSelectionResult {
        // Склады, где есть нужное количество
        $available = WarehouseStock::where('product_id', $productId)
            ->where('available', '>=', $quantity)
            ->with('warehouse')
            ->orderByDesc('warehouse.priority')
            ->get();

        if ($available->isEmpty()) {
            // Попробовать split — взять с нескольких складов
            return $this->splitWarehouseSelection($productId, $quantity);
        }

        // Сортировка по близости к покупателю
        $coords = $this->geocoder->getCoords($destinationCity);

        if ($coords) {
            $sorted = $available->sortBy(function ($stock) use ($coords) {
                return $this->haversineDistance(
                    $coords['lat'], $coords['lng'],
                    $stock->warehouse->lat, $stock->warehouse->lng,
                );
            });

            return new WarehouseSelectionResult(
                warehouse:   $sorted->first()->warehouse,
                isSplit:     false,
            );
        }

        // Геокодирование не удалось — берём по приоритету
        return new WarehouseSelectionResult(
            warehouse: $available->first()->warehouse,
            isSplit:   false,
        );
    }

    private function haversineDistance(float $lat1, float $lng1, float $lat2, float $lng2): float
    {
        $R   = 6371; // km
        $dLat = deg2rad($lat2 - $lat1);
        $dLng = deg2rad($lng2 - $lng1);

        $a = sin($dLat / 2) ** 2
           + cos(deg2rad($lat1)) * cos(deg2rad($lat2)) * sin($dLng / 2) ** 2;

        return $R * 2 * atan2(sqrt($a), sqrt(1 - $a));
    }
}

Расчёт доставки с учётом склада

class RegionalDeliveryCalculator
{
    public function calculate(
        array  $cartItems,
        string $destination,
    ): DeliveryResult {
        // Сгруппировать товары по складу
        $warehouseGroups = $this->groupByWarehouse($cartItems, $destination);

        $allOptions = collect();

        foreach ($warehouseGroups as $warehouseId => $items) {
            $warehouse = Warehouse::find($warehouseId);

            // Считать доставку от конкретного склада
            $request = new DeliveryRequest(
                fromCity:    $warehouse->city,
                fromLat:     $warehouse->lat,
                fromLng:     $warehouse->lng,
                destination: $destination,
                items:       $items,
            );

            $options = $this->carrierCalculator->calculate($request);

            // Если несколько складов — суммировать стоимость
            if (count($warehouseGroups) > 1) {
                $allOptions = $allOptions->merge(
                    $options->map(fn($o) => $o->withWarehouseNote($warehouse->city))
                );
            } else {
                $allOptions = $allOptions->merge($options);
            }
        }

        // Дедупликация по методу доставки — выбрать лучший вариант
        return new DeliveryResult(
            options: $allOptions->groupBy('method')
                ->map(fn($g) => $g->sortBy('price')->first())
                ->values(),
        );
    }

    private function groupByWarehouse(array $cartItems, string $destination): array
    {
        $groups = [];

        foreach ($cartItems as $item) {
            $warehouse = $this->selector->selectForDelivery(
                $item->product_id,
                $item->quantity,
                $destination,
            );

            $warehouseId = $warehouse?->warehouse->id ?? $this->defaultWarehouse->id;
            $groups[$warehouseId][] = $item;
        }

        return $groups;
    }
}

Резервирование стока при заказе

class StockReservationService
{
    public function reserve(Order $order): ReservationResult
    {
        $reservations = [];

        DB::transaction(function () use ($order, &$reservations) {
            foreach ($order->items as $item) {
                // Найти склад с нужным количеством
                $stock = WarehouseStock::where('product_id', $item->product_id)
                    ->where('available', '>=', $item->quantity)
                    ->where('warehouse_id', $item->preferred_warehouse_id
                        ?? $this->getNearestWarehouse($order->delivery_city, $item->product_id)->id
                    )
                    ->lockForUpdate()
                    ->first();

                if (!$stock) {
                    throw new InsufficientStockException(
                        "Недостаточно товара на складе: SKU {$item->product->sku}"
                    );
                }

                // Увеличить зарезервированное
                $stock->increment('reserved', $item->quantity);

                $reservations[] = StockReservation::create([
                    'order_id'     => $order->id,
                    'order_item_id' => $item->id,
                    'warehouse_id' => $stock->warehouse_id,
                    'product_id'   => $item->product_id,
                    'quantity'     => $item->quantity,
                    'expires_at'   => now()->addMinutes(30),
                ]);
            }
        });

        return new ReservationResult(reservations: $reservations);
    }

    public function cancelExpiredReservations(): int
    {
        $expired = StockReservation::where('status', 'reserved')
            ->where('expires_at', '<', now())
            ->get();

        foreach ($expired as $reservation) {
            DB::transaction(function () use ($reservation) {
                WarehouseStock::where([
                    'product_id'  => $reservation->product_id,
                    'warehouse_id' => $reservation->warehouse_id,
                ])->decrement('reserved', $reservation->quantity);

                $reservation->update(['status' => 'cancelled']);
            });
        }

        return $expired->count();
    }
}

Отображение остатков по складам на карточке

// API: наличие по городам
public function stockByCity(int $productId): JsonResponse
{
    $stocks = WarehouseStock::where('product_id', $productId)
        ->where('available', '>', 0)
        ->with('warehouse:id,name,city')
        ->get(['warehouse_id', 'available']);

    return response()->json(
        $stocks->map(fn($s) => [
            'city'      => $s->warehouse->city,
            'warehouse' => $s->warehouse->name,
            'qty'       => $s->available,
        ])
    );
}
// Компонент наличия по городам
const StockByCity: React.FC<{ productId: number }> = ({ productId }) => {
  const { data } = useQuery(['stock', productId], () => fetchStockByCity(productId));

  if (!data?.length) return <span className="text-red-500">Нет в наличии</span>;

  return (
    <details className="text-sm">
      <summary className="cursor-pointer text-green-600 font-medium">
        В наличии — {data.length} город(а)
      </summary>
      <ul className="mt-1 space-y-1 pl-3">
        {data.map(s => (
          <li key={s.warehouse} className="text-gray-600">
            {s.city} — {s.qty} шт.
          </li>
        ))}
      </ul>
    </details>
  );
};

Трансфер между складами

Если нужный товар есть только на дальнем складе, система может предложить трансфер:

CREATE TABLE warehouse_transfers (
    id              BIGSERIAL PRIMARY KEY,
    from_warehouse  BIGINT REFERENCES warehouses(id),
    to_warehouse    BIGINT REFERENCES warehouses(id),
    product_id      BIGINT REFERENCES products(id),
    quantity        INT NOT NULL,
    status          VARCHAR(20) DEFAULT 'pending',
    transit_days    SMALLINT,
    created_at      TIMESTAMP DEFAULT NOW()
);

Сроки реализации

  • Схема данных + WarehouseStock + базовый WarehouseSelector: 2 дня
  • RegionalDeliveryCalculator с учётом координат: 1–2 дня
  • StockReservationService + автоотмена: 1 день
  • API наличия по городам + фронтенд-компонент: 1 день
  • Интерфейс управления складами в админке: 1 день
  • Трансферы между складами: +1–2 дня

Итого без трансферов: 6–7 рабочих дней.