Add out-of-stock detection and display

- Add stock_status column to products table (in_stock/out_of_stock/unknown)
- Detect out-of-stock status on Amazon by checking:
  - #availability text for "currently unavailable"
  - #outOfStock element presence
  - Missing "Add to Cart" button
- Add generic stock status detection for other sites
- Allow adding out-of-stock products (they just won't have a price)
- Update background scheduler to track stock status changes
- Display stock status badge in product list and detail pages
- Dim out-of-stock products in the dashboard
- Show "Currently Unavailable" badge instead of price when out of stock

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
clucraft 2026-01-20 20:54:12 -05:00
parent bf111e13d8
commit 8c5d20707d
9 changed files with 274 additions and 44 deletions

View file

@ -35,6 +35,8 @@ export const userQueries = {
};
// Product types and queries
export type StockStatus = 'in_stock' | 'out_of_stock' | 'unknown';
export interface Product {
id: number;
user_id: number;
@ -43,6 +45,7 @@ export interface Product {
image_url: string | null;
refresh_interval: number;
last_checked: Date | null;
stock_status: StockStatus;
created_at: Date;
}
@ -159,13 +162,14 @@ export const productQueries = {
url: string,
name: string | null,
imageUrl: string | null,
refreshInterval: number = 3600
refreshInterval: number = 3600,
stockStatus: StockStatus = 'unknown'
): Promise<Product> => {
const result = await pool.query(
`INSERT INTO products (user_id, url, name, image_url, refresh_interval)
VALUES ($1, $2, $3, $4, $5)
`INSERT INTO products (user_id, url, name, image_url, refresh_interval, stock_status)
VALUES ($1, $2, $3, $4, $5, $6)
RETURNING *`,
[userId, url, name, imageUrl, refreshInterval]
[userId, url, name, imageUrl, refreshInterval, stockStatus]
);
return result.rows[0];
},
@ -215,6 +219,13 @@ export const productQueries = {
);
},
updateStockStatus: async (id: number, stockStatus: StockStatus): Promise<void> => {
await pool.query(
'UPDATE products SET stock_status = $1 WHERE id = $2',
[stockStatus, id]
);
},
findDueForRefresh: async (): Promise<Product[]> => {
const result = await pool.query(
`SELECT * FROM products

View file

@ -1,7 +1,7 @@
import { Router, Response } from 'express';
import { AuthRequest, authMiddleware } from '../middleware/auth';
import { productQueries, priceHistoryQueries } from '../models';
import { scrapePrice } from '../services/scraper';
import { scrapeProduct } from '../services/scraper';
const router = Router();
@ -62,27 +62,31 @@ router.post('/:productId/refresh', async (req: AuthRequest, res: Response) => {
return;
}
// Scrape new price
const priceData = await scrapePrice(product.url);
// Scrape product data including price and stock status
const scrapedData = await scrapeProduct(product.url);
if (!priceData) {
res.status(400).json({ error: 'Could not extract price from URL' });
return;
// Update stock status
await productQueries.updateStockStatus(productId, scrapedData.stockStatus);
// Record new price if available
let newPrice = null;
if (scrapedData.price) {
newPrice = await priceHistoryQueries.create(
productId,
scrapedData.price.price,
scrapedData.price.currency
);
}
// Record new price
const newPrice = await priceHistoryQueries.create(
productId,
priceData.price,
priceData.currency
);
// Update last_checked timestamp
await productQueries.updateLastChecked(productId);
res.json({
message: 'Price refreshed successfully',
message: scrapedData.stockStatus === 'out_of_stock'
? 'Product is currently out of stock'
: 'Price refreshed successfully',
price: newPrice,
stockStatus: scrapedData.stockStatus,
});
} catch (error) {
console.error('Error refreshing price:', error);

View file

@ -42,28 +42,32 @@ router.post('/', async (req: AuthRequest, res: Response) => {
// Scrape product info
const scrapedData = await scrapeProduct(url);
if (!scrapedData.price) {
// Allow adding out-of-stock products, but require a price for in-stock ones
if (!scrapedData.price && scrapedData.stockStatus !== 'out_of_stock') {
res.status(400).json({
error: 'Could not extract price from the provided URL',
});
return;
}
// Create product
// Create product with stock status
const product = await productQueries.create(
userId,
url,
scrapedData.name,
scrapedData.imageUrl,
refresh_interval || 3600
refresh_interval || 3600,
scrapedData.stockStatus
);
// Record initial price
await priceHistoryQueries.create(
product.id,
scrapedData.price.price,
scrapedData.price.currency
);
// Record initial price if available
if (scrapedData.price) {
await priceHistoryQueries.create(
product.id,
scrapedData.price.price,
scrapedData.price.currency
);
}
// Update last_checked timestamp
await productQueries.updateLastChecked(product.id);

View file

@ -1,6 +1,6 @@
import cron from 'node-cron';
import { productQueries, priceHistoryQueries } from '../models';
import { scrapePrice } from './scraper';
import { scrapeProduct } from './scraper';
let isRunning = false;
@ -22,25 +22,35 @@ async function checkPrices(): Promise<void> {
try {
console.log(`Checking price for product ${product.id}: ${product.url}`);
const priceData = await scrapePrice(product.url);
const scrapedData = await scrapeProduct(product.url);
if (priceData) {
// Update stock status
if (scrapedData.stockStatus !== product.stock_status) {
await productQueries.updateStockStatus(product.id, scrapedData.stockStatus);
console.log(
`Stock status changed for product ${product.id}: ${product.stock_status} -> ${scrapedData.stockStatus}`
);
}
if (scrapedData.price) {
// Get the latest recorded price to compare
const latestPrice = await priceHistoryQueries.getLatest(product.id);
// Only record if price has changed or it's the first entry
if (!latestPrice || latestPrice.price !== priceData.price) {
if (!latestPrice || latestPrice.price !== scrapedData.price.price) {
await priceHistoryQueries.create(
product.id,
priceData.price,
priceData.currency
scrapedData.price.price,
scrapedData.price.currency
);
console.log(
`Recorded new price for product ${product.id}: ${priceData.currency} ${priceData.price}`
`Recorded new price for product ${product.id}: ${scrapedData.price.currency} ${scrapedData.price.price}`
);
} else {
console.log(`Price unchanged for product ${product.id}`);
}
} else if (scrapedData.stockStatus === 'out_of_stock') {
console.log(`Product ${product.id} is out of stock, no price available`);
} else {
console.warn(`Could not extract price for product ${product.id}`);
}

View file

@ -6,17 +6,20 @@ import {
findMostLikelyPrice,
} from '../utils/priceParser';
export type StockStatus = 'in_stock' | 'out_of_stock' | 'unknown';
export interface ScrapedProduct {
name: string | null;
price: ParsedPrice | null;
imageUrl: string | null;
url: string;
stockStatus: StockStatus;
}
// Site-specific scraper configurations
interface SiteScraper {
match: (url: string) => boolean;
scrape: ($: CheerioAPI, url: string) => Partial<ScrapedProduct>;
scrape: ($: CheerioAPI, url: string) => Partial<Omit<ScrapedProduct, 'url'>>;
}
const siteScrapers: SiteScraper[] = [
@ -133,7 +136,38 @@ const siteScrapers: SiteScraper[] = [
$('img[data-a-dynamic-image]').attr('src') ||
null;
return { name, price, imageUrl };
// Stock status detection
let stockStatus: StockStatus = 'unknown';
const availabilityText = $('#availability').text().toLowerCase();
const outOfStockDiv = $('#outOfStock').length > 0;
const unavailableText = $('body').text().toLowerCase();
// Check for out of stock indicators
if (
outOfStockDiv ||
availabilityText.includes('currently unavailable') ||
availabilityText.includes('out of stock') ||
availabilityText.includes('not available') ||
$('#add-to-cart-button').length === 0 && $('#buy-now-button').length === 0
) {
// Verify it's truly out of stock by checking for unavailable messaging
if (
unavailableText.includes('currently unavailable') ||
unavailableText.includes("we don't know when or if this item will be back in stock") ||
outOfStockDiv ||
availabilityText.includes('out of stock')
) {
stockStatus = 'out_of_stock';
}
} else if (
availabilityText.includes('in stock') ||
availabilityText.includes('available') ||
$('#add-to-cart-button').length > 0
) {
stockStatus = 'in_stock';
}
return { name, price, imageUrl, stockStatus };
},
},
@ -407,6 +441,7 @@ export async function scrapeProduct(url: string): Promise<ScrapedProduct> {
price: null,
imageUrl: null,
url,
stockStatus: 'unknown',
};
try {
@ -442,6 +477,7 @@ export async function scrapeProduct(url: string): Promise<ScrapedProduct> {
if (siteResult.name) result.name = siteResult.name;
if (siteResult.price) result.price = siteResult.price;
if (siteResult.imageUrl) result.imageUrl = siteResult.imageUrl;
if (siteResult.stockStatus) result.stockStatus = siteResult.stockStatus;
}
// Try JSON-LD structured data
@ -467,6 +503,11 @@ export async function scrapeProduct(url: string): Promise<ScrapedProduct> {
result.imageUrl = extractGenericImage($, url);
}
// Generic stock status detection if not already set
if (result.stockStatus === 'unknown') {
result.stockStatus = extractGenericStockStatus($);
}
// Try Open Graph meta tags as last resort
if (!result.name) {
result.name = $('meta[property="og:title"]').attr('content') || null;
@ -635,6 +676,68 @@ function extractGenericImage($: CheerioAPI, baseUrl: string): string | null {
return null;
}
function extractGenericStockStatus($: CheerioAPI): StockStatus {
const bodyText = $('body').text().toLowerCase();
// Common out-of-stock indicators
const outOfStockPatterns = [
'out of stock',
'currently unavailable',
'not available',
'sold out',
'no longer available',
'temporarily out of stock',
'unavailable',
'back in stock soon',
'notify me when available',
'out-of-stock',
];
// Common in-stock indicators
const inStockPatterns = [
'in stock',
'add to cart',
'add to basket',
'buy now',
'available',
'ships from',
'ready to ship',
];
// Check for out-of-stock patterns
for (const pattern of outOfStockPatterns) {
if (bodyText.includes(pattern)) {
// Make sure it's not negated (e.g., "not out of stock")
const index = bodyText.indexOf(pattern);
const before = bodyText.slice(Math.max(0, index - 10), index);
if (!before.includes('not ') && !before.includes("isn't ") && !before.includes("won't be ")) {
return 'out_of_stock';
}
}
}
// Check for in-stock indicators
for (const pattern of inStockPatterns) {
if (bodyText.includes(pattern)) {
return 'in_stock';
}
}
// Check for schema.org availability
const availability = $('[itemprop="availability"]').attr('content') ||
$('[itemprop="availability"]').attr('href') || '';
if (availability.toLowerCase().includes('outofstock') ||
availability.toLowerCase().includes('discontinued')) {
return 'out_of_stock';
}
if (availability.toLowerCase().includes('instock') ||
availability.toLowerCase().includes('available')) {
return 'in_stock';
}
return 'unknown';
}
export async function scrapePrice(url: string): Promise<ParsedPrice | null> {
const product = await scrapeProduct(url);
return product.price;