Backend:
- Migrations 013-015: eBay user tokens, monitor settings on saved_searches
(monitor_enabled, poll_interval_min, min_trust_score, last_checked_at),
watch_alerts table with UNIQUE dedup on (saved_search_id, platform_listing_id),
active_monitors registry for cross-user polling
- WatchAlert model + store methods: upsert_alert, list_alerts, dismiss_alert,
count_undismissed_alerts, dismiss_all_alerts, list_active_monitors
- monitor.py: run_monitor_search() using TrustScorer.score_batch(); should_alert()
with BIN/auction/partial-score logic (auction window = 24h, partial +10 buffer)
- PATCH /api/saved-searches/{id}/monitor, GET /api/alerts, POST /api/alerts/*/dismiss
- Background polling loop at startup (asyncio.to_thread every 60s check cycle)
- ebay/adapter.py: enrich_seller_trading_api() via Trading API GetUser (OAuth token)
- nginx: raise proxy_read_timeout to 120s for slow eBay search responses
Frontend:
- AlertBell component: bell button + unread badge + panel with dismiss/clear-all;
polls /api/alerts every 2 minutes; aria-live announcement on count change
- alerts.ts Pinia store: fetchAlerts, dismiss, dismissAll
- SavedSearchesView: monitor toggle + poll interval + min trust score controls
- SettingsView: eBay OAuth connect/disconnect section
- AppNav: AlertBell wired for logged-in and local-tier users
Tests: 24 monitor tests (should_alert branches, store alert CRUD, run_monitor_search
with mocked adapter); fix browser_pool test assertions for new wait_for_* params.
24 lines
1.1 KiB
SQL
24 lines
1.1 KiB
SQL
-- Migration 014: background monitor settings on saved_searches + watch_alerts table
|
|
|
|
ALTER TABLE saved_searches ADD COLUMN monitor_enabled INTEGER NOT NULL DEFAULT 0;
|
|
ALTER TABLE saved_searches ADD COLUMN poll_interval_min INTEGER NOT NULL DEFAULT 60;
|
|
ALTER TABLE saved_searches ADD COLUMN min_trust_score INTEGER NOT NULL DEFAULT 60;
|
|
ALTER TABLE saved_searches ADD COLUMN last_checked_at TEXT;
|
|
|
|
CREATE TABLE IF NOT EXISTS watch_alerts (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
saved_search_id INTEGER NOT NULL REFERENCES saved_searches(id) ON DELETE CASCADE,
|
|
platform_listing_id TEXT NOT NULL,
|
|
title TEXT NOT NULL,
|
|
price REAL NOT NULL,
|
|
currency TEXT NOT NULL DEFAULT 'USD',
|
|
trust_score INTEGER NOT NULL,
|
|
url TEXT,
|
|
first_alerted_at TEXT NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
|
dismissed_at TEXT,
|
|
UNIQUE(saved_search_id, platform_listing_id)
|
|
);
|
|
|
|
CREATE INDEX IF NOT EXISTS idx_watch_alerts_undismissed
|
|
ON watch_alerts(saved_search_id)
|
|
WHERE dismissed_at IS NULL;
|