Some checks failed
CI / Frontend Lint & Type Check (push) Has been cancelled
CI / Frontend Build (push) Has been cancelled
CI / Backend Lint (push) Has been cancelled
CI / Backend Tests (push) Has been cancelled
CI / Docker Build (push) Has been cancelled
CI / Security Scan (push) Has been cancelled
Deploy / Build & Push Images (push) Has been cancelled
Deploy / Deploy to Server (push) Has been cancelled
Deploy / Notify (push) Has been cancelled
## Watchlist & Monitoring - ✅ Automatic domain monitoring based on subscription tier - ✅ Email alerts when domains become available - ✅ Health checks (DNS/HTTP/SSL) with caching - ✅ Expiry warnings for domains <30 days - ✅ Weekly digest emails - ✅ Instant alert toggle (optimistic UI updates) - ✅ Redesigned health check overlays with full details - 🔒 'Not public' display for .ch/.de domains without public expiry ## Portfolio Management (NEW) - ✅ Track owned domains with purchase price & date - ✅ ROI calculation (unrealized & realized) - ✅ Domain valuation with auto-refresh - ✅ Renewal date tracking - ✅ Sale recording with profit calculation - ✅ List domains for sale directly from portfolio - ✅ Full portfolio summary dashboard ## Listings / For Sale - ✅ Renamed from 'Portfolio' to 'For Sale' - ✅ Fixed listing limits: Scout=0, Trader=5, Tycoon=50 - ✅ Featured badge for Tycoon listings - ✅ Inquiries modal for sellers - ✅ Email notifications when buyer inquires - ✅ Inquiries column in listings table ## Scrapers & Data - ✅ Added 4 new registrar scrapers (Namecheap, Cloudflare, GoDaddy, Dynadot) - ✅ Increased scraping frequency to 2x daily (03:00 & 15:00 UTC) - ✅ Real historical data from database - ✅ Fixed RDAP/WHOIS for .ch/.de domains - ✅ Enhanced SSL certificate parsing ## Scheduler Jobs - ✅ Tiered domain checks (Scout=daily, Trader=hourly, Tycoon=10min) - ✅ Daily health checks (06:00 UTC) - ✅ Weekly expiry warnings (Mon 08:00 UTC) - ✅ Weekly digest emails (Sun 10:00 UTC) - ✅ Auction cleanup every 15 minutes ## UI/UX Improvements - ✅ Removed 'Back' buttons from Intel pages - ✅ Redesigned Radar page to match Market/Intel design - ✅ Less prominent check frequency footer - ✅ Consistent StatCard components across all pages - ✅ Ambient background glows - ✅ Better error handling ## Documentation - ✅ Updated README with monitoring section - ✅ Added env.example with all required variables - ✅ Updated Memory Bank (activeContext.md) - ✅ SMTP configuration requirements documented
107 lines
4.2 KiB
Python
107 lines
4.2 KiB
Python
"""Cloudflare Registrar TLD price scraper."""
|
|
import logging
|
|
from datetime import datetime
|
|
from typing import Optional
|
|
|
|
import httpx
|
|
|
|
from app.services.tld_scraper.base import BaseTLDScraper, TLDPriceData, ScraperError
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class CloudflareScraper(BaseTLDScraper):
|
|
"""
|
|
Scraper for Cloudflare Registrar domain prices.
|
|
|
|
Cloudflare sells domains at-cost (wholesale price), so their prices
|
|
are often the lowest available and serve as a baseline.
|
|
|
|
Note: Cloudflare doesn't have a public API, but we can use their
|
|
known at-cost pricing which they publish.
|
|
"""
|
|
|
|
name = "cloudflare"
|
|
base_url = "https://www.cloudflare.com/products/registrar/"
|
|
|
|
# Cloudflare prices are at-cost (wholesale).
|
|
# These prices are well-documented and rarely change.
|
|
# Source: https://www.cloudflare.com/products/registrar/
|
|
CLOUDFLARE_PRICES = {
|
|
# Major TLDs (at wholesale cost)
|
|
"com": {"reg": 10.44, "renew": 10.44, "transfer": 10.44},
|
|
"net": {"reg": 11.94, "renew": 11.94, "transfer": 11.94},
|
|
"org": {"reg": 10.11, "renew": 10.11, "transfer": 10.11},
|
|
"info": {"reg": 11.44, "renew": 11.44, "transfer": 11.44},
|
|
"biz": {"reg": 13.44, "renew": 13.44, "transfer": 13.44},
|
|
"co": {"reg": 11.02, "renew": 11.02, "transfer": 11.02},
|
|
"io": {"reg": 33.98, "renew": 33.98, "transfer": 33.98},
|
|
"me": {"reg": 14.94, "renew": 14.94, "transfer": 14.94},
|
|
"dev": {"reg": 11.94, "renew": 11.94, "transfer": 11.94},
|
|
"app": {"reg": 14.94, "renew": 14.94, "transfer": 14.94},
|
|
"xyz": {"reg": 10.44, "renew": 10.44, "transfer": 10.44},
|
|
|
|
# ccTLDs supported by Cloudflare
|
|
"uk": {"reg": 8.50, "renew": 8.50, "transfer": 8.50},
|
|
"de": {"reg": 7.05, "renew": 7.05, "transfer": 7.05},
|
|
"eu": {"reg": 9.00, "renew": 9.00, "transfer": 9.00},
|
|
"nl": {"reg": 9.20, "renew": 9.20, "transfer": 9.20},
|
|
"ca": {"reg": 12.42, "renew": 12.42, "transfer": 12.42},
|
|
"fr": {"reg": 10.22, "renew": 10.22, "transfer": 10.22},
|
|
"es": {"reg": 10.05, "renew": 10.05, "transfer": 10.05},
|
|
"it": {"reg": 10.99, "renew": 10.99, "transfer": 10.99},
|
|
|
|
# New gTLDs
|
|
"club": {"reg": 11.94, "renew": 11.94, "transfer": 11.94},
|
|
"shop": {"reg": 28.94, "renew": 28.94, "transfer": 28.94},
|
|
"blog": {"reg": 25.94, "renew": 25.94, "transfer": 25.94},
|
|
"site": {"reg": 25.94, "renew": 25.94, "transfer": 25.94},
|
|
"live": {"reg": 21.94, "renew": 21.94, "transfer": 21.94},
|
|
"cloud": {"reg": 19.94, "renew": 19.94, "transfer": 19.94},
|
|
}
|
|
|
|
async def scrape(self) -> list[TLDPriceData]:
|
|
"""
|
|
Return Cloudflare's known at-cost pricing.
|
|
|
|
Cloudflare doesn't have a public API for pricing, but their
|
|
prices are well-documented and stable (at wholesale cost).
|
|
|
|
Returns:
|
|
List of TLDPriceData objects with Cloudflare pricing
|
|
"""
|
|
results = []
|
|
now = datetime.utcnow()
|
|
|
|
for tld, prices in self.CLOUDFLARE_PRICES.items():
|
|
results.append(TLDPriceData(
|
|
tld=tld,
|
|
registrar="cloudflare",
|
|
registration_price=prices["reg"],
|
|
renewal_price=prices["renew"],
|
|
transfer_price=prices.get("transfer"),
|
|
currency="USD",
|
|
source="static", # These are known prices, not scraped
|
|
confidence=1.0, # At-cost pricing is reliable
|
|
scraped_at=now,
|
|
notes="At-cost (wholesale) pricing",
|
|
))
|
|
|
|
logger.info(f"Loaded {len(results)} Cloudflare at-cost prices")
|
|
return results
|
|
|
|
async def health_check(self) -> bool:
|
|
"""Check if Cloudflare is accessible."""
|
|
try:
|
|
async with httpx.AsyncClient(timeout=10.0) as client:
|
|
response = await client.get(
|
|
self.base_url,
|
|
headers=self.get_headers(),
|
|
follow_redirects=True,
|
|
)
|
|
return response.status_code == 200
|
|
except Exception as e:
|
|
logger.debug(f"Cloudflare health check failed: {e}")
|
|
return False
|
|
|