inventory-tracking
Inventory Tracking
Overview
Real-time inventory tracking prevents overselling by reserving stock when customers add items to their cart and decrementing it on order fulfillment. Every major e-commerce platform has this built in. Platform-native inventory tracking is almost always the right starting point — only build custom inventory logic if you have requirements that platforms cannot meet (complex multi-warehouse routing, custom reservation windows, or external WMS integration).
When to Use This Skill
- When overselling is occurring and orders are being placed for out-of-stock items
- When implementing multi-warehouse inventory with per-location stock levels
- When building backorder or pre-order functionality for out-of-stock products
- When a flash sale or product launch will create high-concurrency checkout attempts for limited-stock items
Core Instructions
Step 1: Determine platform and enable inventory tracking
| Platform | Built-in Inventory | Recommended Extension |
|---|---|---|
| Shopify | Native per-variant inventory tracking with location support | Stocky (free, by Shopify) for purchase orders and demand forecasting |
| WooCommerce | Native stock management with backorder support | ATUM Inventory Management for advanced multi-warehouse and supplier POs |
| BigCommerce | Native per-SKU inventory tracking with low-stock alerts | Multi-Location Inventory app for warehouse routing |
| Custom / Headless | Build atomic reservation with optimistic locking | Required for custom platforms without native inventory management |
Step 2: Platform-specific setup
Shopify
Shopify tracks inventory per variant, per location, natively.
Enable inventory tracking:
- Go to Admin → Products → [Product] → [Variant]
- Under Inventory, check Track quantity
- Enter your quantity per location
- For "Continue selling when out of stock" — only check this if you allow backorders for this product
Set up locations:
- Go to Settings → Locations
- Add each warehouse, store, or fulfillment center as a location
- When editing a product variant, set the quantity at each location independently
Backorders:
- Enable per variant: check Continue selling when out of stock on the variant
- Or use a backorder app like Pre-Order Now or Back In Stock for more control (notify customers when available, collect pre-orders, etc.)
Oversell prevention during high traffic:
Shopify's checkout system holds inventory during the checkout process to prevent two customers from purchasing the last item simultaneously. For flash sales, use Shopify Scripts (Plus) or the Inventory Planner app to set purchase limits.
Inventory sync with physical locations:
- Install Stocky (free, by Shopify) for purchase orders and receiving
- Stocky automatically updates Shopify inventory when you receive a PO
- Use the Shopify POS app to sync inventory between your online store and physical retail locations
WooCommerce
WooCommerce has built-in stock management.
Enable inventory tracking:
- Go to WooCommerce → Settings → Products → Inventory
- Check Enable stock management
- Set Hold stock (minutes) — this is the inventory reservation window during checkout (default: 60 minutes)
Per-product settings:
- Go to WooCommerce → Products → [Product] → Inventory tab
- Enable Manage stock?
- Enter Stock quantity
- Set Backorders: "Do not allow" / "Allow, but notify customer" / "Allow"
- Set a Low stock threshold for this product
Advanced inventory management with ATUM:
- Install ATUM Inventory Management for WooCommerce (free core, paid advanced features)
- ATUM provides a central inventory dashboard showing stock levels across all products
- Add purchase orders through ATUM → Purchase Orders → Add PO
- Receiving a PO in ATUM automatically increments WooCommerce stock
- For multi-warehouse: ATUM's Multi-Inventory add-on ($) assigns stock per location and routes fulfillment
BigCommerce
BigCommerce tracks inventory per SKU natively.
Enable inventory tracking:
- Go to Products → [Product] → Inventory tab
- Set Inventory tracking to "By product" or "By option" (for variants)
- Enter the current stock level
- Set a Low stock level for alerts
Multi-location inventory:
- Install the Multi-Location Inventory app from the BigCommerce App Marketplace
- Assign stock quantities per location
- Set fulfillment routing rules (ship from closest, ship from cheapest, etc.)
Backorders:
- BigCommerce handles this natively — set Allow backorders on products you want to continue selling when out of stock
- The product page shows "Ships in X days" when on backorder
Custom / Headless
For custom platforms, implement atomic inventory reservation using optimistic concurrency control to prevent overselling under high load:
// lib/inventory.ts
const MAX_RETRIES = 3;
// Reserve inventory atomically — handles concurrent requests safely
export async function reserveInventory({
variantId, locationId, quantity, referenceId
}: { variantId: string; locationId: string; quantity: number; referenceId: string }) {
for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
const level = await db.inventoryLevels.findUnique({
where: { variantId_locationId: { variantId, locationId } },
});
if (!level) throw new Error(`Inventory not found: ${variantId}`);
const available = level.onHand - level.reserved;
if (available < quantity && !level.backorderAllowed) {
throw new Error(`Insufficient stock: ${available} available, ${quantity} requested`);
}
// Optimistic update — only succeeds if version hasn't changed (no concurrent modifications)
const updated = await db.inventoryLevels.updateMany({
where: { variantId_locationId: { variantId, locationId }, version: level.version },
data: { reserved: level.reserved + quantity, version: level.version + 1 },
});
if (updated.count === 0) {
// Another process modified inventory concurrently; retry
await new Promise(r => setTimeout(r, 50 * (attempt + 1)));
continue;
}
// Log the transaction for audit trail
await db.inventoryTransactions.create({
data: { variantId, locationId, type: 'reserve', quantity: -quantity, referenceId },
});
return { success: true, remaining: available - quantity };
}
throw new Error(`Failed to reserve inventory after ${MAX_RETRIES} retries`);
}
// Release reservation when cart expires or order is cancelled
export async function releaseReservation({
variantId, locationId, quantity, referenceId
}: { variantId: string; locationId: string; quantity: number; referenceId: string }) {
// Idempotency check — don't release twice
const existing = await db.inventoryTransactions.findFirst({
where: { type: 'release', referenceId, variantId },
});
if (existing) return;
await db.$transaction([
db.inventoryLevels.update({
where: { variantId_locationId: { variantId, locationId } },
data: { reserved: { decrement: quantity } },
}),
db.inventoryTransactions.create({
data: { variantId, locationId, type: 'release', quantity: +quantity, referenceId },
}),
]);
}
// Expire stale cart reservations — run every 5-10 minutes via cron
export async function expireStaleCartReservations() {
const TTL_MINUTES = 30;
const cutoff = new Date(Date.now() - TTL_MINUTES * 60 * 1000);
const staleCarts = await db.carts.findMany({
where: { status: 'active', updatedAt: { lt: cutoff }, reservedAt: { not: null } },
include: { items: true },
});
for (const cart of staleCarts) {
for (const item of cart.items) {
await releaseReservation({ variantId: item.variantId, locationId: item.locationId, quantity: item.quantity, referenceId: cart.id });
}
}
}
Step 3: Configure backorder behavior
Backorders allow customers to purchase even when stock is at zero, with a clear expectation of a delayed delivery.
When to allow backorders:
- Products with reliable supplier lead times (7–14 days)
- Made-to-order products
- Pre-order campaigns for upcoming products
When to block backorders:
- Products with unreliable supply
- Third-party fulfilled items where you don't control restock
Communication best practices:
- Show "Ships in 7–10 days" on the product page when stock is 0 and backorders are enabled
- Include expected ship date in the order confirmation email
- Notify customers proactively if the expected date changes
Step 4: Set up inventory alerts
Pair inventory tracking with low-stock alerts — see the @low-stock-alerts skill for full setup. Quick summary:
- Shopify: Go to Admin → Products — Shopify shows a low stock indicator; use Stocky for email alerts
- WooCommerce: Go to WooCommerce → Settings → Products → Inventory → Low stock threshold — WooCommerce emails the store admin when stock crosses this threshold
- BigCommerce: Go to Products → [Product] → Inventory → Low stock level — BigCommerce sends email notifications automatically
Best Practices
- Enable inventory tracking on every SKU — products without tracking can be oversold silently; only disable tracking for digital products or items with unlimited supply
- Set a reservation window for in-progress checkouts — WooCommerce's "Hold stock" setting (default 60 min) prevents inventory from being held indefinitely by abandoned carts
- Test your oversell protection — during a flash sale setup, manually verify that the last unit cannot be purchased twice by simulating two simultaneous checkouts
- Log every inventory change — platforms log this natively; for custom builds, an immutable
inventory_transactionstable is essential for diagnosing discrepancies - Use the platform's built-in multi-location inventory before buying a third-party app — Shopify, WooCommerce, and BigCommerce all handle multiple locations natively
Common Pitfalls
| Problem | Solution |
|---|---|
| Overselling during flash sales on Shopify | Shopify's checkout holds inventory during the checkout flow; for very high-concurrency launches, set purchase limits using Shopify Scripts (Plus) or use a waitlist app |
| WooCommerce inventory not decremented after order | Check that stock management is enabled on the product AND globally in WooCommerce settings; both must be on |
| Inventory released immediately when order is cancelled before fulfillment | This is correct behavior for physical goods — released inventory becomes available for other customers; only delay release for backordered items |
| Negative inventory after manual adjustment | Add validation in WooCommerce (ATUM) or set a DB check constraint on custom builds; reserved >= 0 and on_hand >= 0 |
| Shopify location not receiving inventory updates from POS | Ensure POS is connected to the correct Shopify location in Settings → Locations → POS channel |
Related Skills
- @multi-warehouse
- @low-stock-alerts
- @variant-matrix