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

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.

Розробка та обслуговування будь-яких видів сайтів:

Інформаційні сайти або веб-програми
Сайти візитки, 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()) {
            // Спробувати розділення — взяти з кількох складів
            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; // км
        $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 робочих днів.